System V IPC(3)-共用記憶體

来源:http://www.cnblogs.com/yuuyuu/archive/2016/01/16/5136467.html
-Advertisement-
Play Games

一.概述 1.共用記憶體允許多個進程共用物理記憶體的同一塊記憶體區。2.與管道和消息隊列不同,共用記憶體在用戶記憶體空間,不需要內核介入。降低了內核和用戶緩衝區的數據複製開銷。所以這種IPC速度比較快。3.多個進程共用記憶體時需要其他同步機制來控制臨界區,如上一篇...


一.概述                                                   

1.共用記憶體允許多個進程共用物理記憶體的同一塊記憶體區。

2.與管道和消息隊列不同,共用記憶體在用戶記憶體空間,不需要內核介入。降低了內核和用戶緩衝區的數據複製開銷。所以這種IPC速度比較快

3.多個進程共用記憶體時需要其他同步機制來控制臨界區,如上一篇的信號量

二.函數介面                                            

1.創建或打開共用記憶體

1 #include <sys/shm.h>
2 
3 int shmget(key_t key, size_t size, int shmflg);

key和shmflg同消息隊列和信號量一樣,這裡不再敘述。

size:需要分配記憶體的大小

2.使用共用記憶體

1 #include <sys/shm.h>
2 
3 void *shmat(int shmid, const void *shmaddr, int shmflg);

shmid:shmget()返回的值

shmaddr:把剛剛創建的記憶體指向該指針。如果是NULL,內核會自動選擇。

shmflg:如果shmaddr不為NULL,shmflg控制shmaddr如果指向記憶體。如記憶體只讀,計算記憶體地址倍數等,可以查看man手冊。

3.分離共用記憶體

1 #include <sys/shm.h>
2 
3 
4 int shmdt(const void *shmaddr);

shmaddr是該進程指向共用記憶體的指針。把該共用記憶體從該進程分離,即該共用記憶體和該進程沒有聯繫了。分離後,共存記憶體依然存在。

4.控制共用記憶體

1 #include <sys/shm.h>
2 
3 int shmctl(int shmid, int cmd, struct shmid_ds *buf);

cmd和消息隊列一樣,IPC_RMID是刪除共用記憶體。

三.簡單例子                                            

我們寫2個小程式,第一個創建和拷貝數據到共用記憶體,第二個讀取共用記憶體數據並刪除共用記憶體。

1.創建和拷貝

 1 /**
 2  * @file shm1.c
 3  */
 4 
 5 #include <stdio.h>
 6 #include <stdlib.h>
 7 #include <string.h>
 8 #include <sys/shm.h>
 9 
10 #define SHARE_MEM_BUF_SIZE 1024
11 
12 void err_exit(const char *err_msg)
13 {
14     printf("error:%s\n", err_msg);
15     exit(1);
16 }
17 
18 int main(void)
19 {
20     int shm_id;
21     void *share_mem;
22     char *text = "123456";
23 
24     shm_id = shmget(IPC_PRIVATE, SHARE_MEM_BUF_SIZE, 0666 | IPC_CREAT);
25     if (shm_id == -1)
26         err_exit("shmget()");
27 
28     printf("shm_id:%d\n", shm_id);
29 
30     /* 把創建的共用記憶體指向該程式的一個指針 */
31     share_mem = shmat(shm_id, NULL, 0);
32     if (share_mem == (void *)-1)
33         err_exit("shmat()");
34 
35     /* 拷貝數據到共用記憶體 */
36     memcpy((char *)share_mem, text, strlen(text));
37 
38     /* 分離共用記憶體 */
39     if (shmdt(share_mem) == -1)
40         err_exit("shmdt()");
41 
42     return 0;
43 }

2.讀取並刪除

 1 /**
 2  * @file shm2.c
 3  */
 4 
 5 #include <stdio.h>
 6 #include <stdlib.h>
 7 #include <string.h>
 8 #include <sys/shm.h>
 9 
10 #define SHARE_MEM_BUF_SIZE 1024
11 
12 void err_exit(const char *err_msg)
13 {
14     printf("error:%s\n", err_msg);
15     exit(1);
16 }
17 
18 int main(int argc, const char *argv[])
19 {
20     if (argc < 2)
21     {
22         printf("usage:%s shm_id\n", argv[0]);
23         exit(1);
24     }
25 
26     void *share_mem;
27     int shm_id = atoi(argv[1]);
28 
29     /* 把創建的共用記憶體指向該程式的一個指針 */
30     share_mem = shmat(shm_id, NULL, 0);
31     if (share_mem == (void *)-1)
32         err_exit("shmat()");
33 
34     /* 讀取數據 */
35     printf("read data:%s\n", (char *)share_mem);
36 
37     /* 分離共用記憶體 */
38     if (shmdt(share_mem) == -1)
39         err_exit("shmdt()");
40 
41     /* 刪除共用記憶體 */
42     if (shmctl(shm_id, IPC_RMID, 0) == -1)
43         err_exit("shmctl()");
44 
45     return 0;
46 }

四.實驗                                                   

1.shm1.c用IPC_PRIVATE方式讓內核自動創建一個key,並分配1024位元組記憶體。第36行把"123456"拷貝到共用記憶體。

2.shm2.c從命令行來指定要使用的共用記憶體,第35行讀取數據,第42行刪除數據。

3.運行shm1後,可以得到共用記憶體的shmid,可以用ipcs -m | grep 'xxx'查看:

可以看到我們分配的1024位元組記憶體。

4.運行shm2接收shmid=50954258的數據並刪除共用記憶體,再用ipcs -m | grep 'xxx'查看:

可以看到讀取了"123456"數據,用ipcs查看時,已被刪除。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Android中進程生命周期的優先順序
  • Android系統集成了一個輕量級的資料庫:SQLite,所以Android對資料庫的支持很好,每個應用都可以方便的使用它。SQLite作為一個嵌入式的資料庫引擎,專門適用於資源有限的設備上適量數據存取,現在的主流移動設備像Android、iPhone等都使用SQLite作為複雜數據的存儲引擎,.....
  • 一、什麼是存儲過程簡單的說,就是一組SQL語句集,功能強大,可以實現一些比較複雜的邏輯功能,類似於JAVA語言中的方法;ps:存儲過程跟觸發器有點類似,都是一組SQL集,但是存儲過程是主動調用的,且功能比觸發器更加強大,觸發器是某件事觸發後自動調用;二、有哪些特性有輸入輸出參數,可以聲明變數,有if...
  • 原文鏈接:http://dblab.xmu.edu.cn/blog/install-hadoop/當開始著手實踐 Hadoop 時,安裝 Hadoop 往往會成為新手的一道門檻。儘管安裝其實很簡單,書上有寫到,官方網站也有 Hadoop 安裝配置教程,但由於對 Linux 環境不熟悉,書上跟官網上簡...
  • 就這段時間,很多人在抱怨為什麼自己的MySQL又打不開問題。 就“Windows(7)上不能啟動MySQL服務(位於本地電腦上)錯誤1067 :進程意外終止”這個問題,我想到了幾種方案解決: 一、首先在windows的服務中確定mysql服務是否開啟, 如果你在windows的服務中找不到my.....
  • 在資料庫中,對數據進行比對,有兩種用法,第一種:不對任何列進行條件判斷SELECT Student_ID AS 學號,English AS 英語成績,CASE WHEN English>=120 THEN '優秀'WHEN English>=60 AND English<=100 THEN '及格....
  • Homebrew OS X 不可或缺的套件管理器,可以說Homebrew就是mac下的apt-get、yum.1.安裝homebrew brew的安裝很簡單,使用一條ruby命令即可,Mac系統上已經預設安裝了ruby。ruby -e "$(curl -fsSL https://raw.git...
  • 系統信息arch 顯示機器的處理器架構(1)uname -m 顯示機器的處理器架構(2)uname -r 顯示正在使用的內核版本dmidecode -q 顯示硬體系統部件 - (SMBIOS / DMI)hdparm -i /dev/hda 羅列一個磁碟的架構特性hdparm -tT /dev/sd...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...