IT培训机构|91免费精品视频|专注编程培训|91免费精品|软件开发培训_91免费国产视频_华清远见教育


linux 共享內存

分享到:
           

    本文關鍵字: linux 共享內存,linux 共享內存編程

    可以說,共享內存是一種為高效的進程間通信方式,因為進程可以直接讀寫內存,不需要任何數據的復制。為了在多個進程間交換信息,內核專門留出了一塊內存區,這段內存區可以由需要訪問的進程將其映射到自己的私有地址空間。因此,進程就可以直接讀寫這一內存區而不需要進行數據的復制,從而大大提高了效率。當然,由于多個進程共享一段內存,因此也需要依靠某種同步機制,如互斥鎖和信號量等。其原理示意圖如圖1所示。


圖1 共享內存原理示意圖

    共享內存的實現分為兩個步驟:第一步是創建共享內存,這里用到的函數是shmget(),也就是從內存中獲得一段共享內存區域;第二步是映射共享內存,也就是把這段創建的共享內存映射到具體的進程空間中,這里使用的函數是shmat()。到這里,就可以使用這段共享內存了,也就是可以使用不帶緩沖的I/O讀寫命令對其進行操作。除此之外,還有撤銷映射的操作,其函數為shmdt()。這里主要介紹這3個函數。

    表1 列舉了shmget()函數的語法要點。

表1 shmget()函數語法要點

所需頭文件 #include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函數原型 int shmget(key_t key, int size, int shmflg)
函數傳入值 key:共享內存的鍵值,多個進程可以通過它訪問同一個共享內存,其中有個特殊值IPC_PRIVATE,用于創建當前進程的私有共享內存
size:共享內存區大小
shmflg:同open()函數的權限位,也可以用八進制表示法
函數返回值 成功:共享內存段標識符
出錯:-1

    表2列舉了shmat()函數的語法要點。

表2 shmat()函數語法要點

所需頭文件 #include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函數原型 char *shmat(int shmid, const void *shmaddr, int shmflg)
函數傳入值 shmid:要映射的共享內存區標識符
shmaddr:將共享內存映射到指定地址(若為0則表示系統自動分配地址并把該段共享內存映射到調用進程的地址空間)
shmflg SHM_RDONLY:共享內存只讀
默認0:共享內存可讀寫
函數返回值 成功:被映射的段地址
出錯:-1

    表3列舉了shmdt()函數的語法要點。

表3 shmdt()函數語法要點

所需頭文件 #include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函數原型 int shmdt(const void *shmaddr)
函數傳入值 shmaddr:被映射的共享內存段地址
函數返回值 成功:0
出錯:-1

    以下實例說明了如何使用基本的共享內存函數。首先創建一個共享內存區(采用的共享內存的鍵值為IPC_PRIVATE,是因為本實例中創建的共享內存是父子進程間的共用部分),然后創建子進程,在父子兩個進程中將共享內存分別映射到各自的進程地址空間中。

    父進程先等待用戶輸入,然后將用戶輸入的字符串寫入到共享內存,之后向共享內存的頭部寫入“WROTE”字符串表示父進程已成功寫入數據。子進程一直等到共享內存的頭部字符串為“WROTE”,然后將共享內存的有效數據(在父進程中用戶輸入的字符串)在屏幕上打印。父子兩個進程在完成以上工作后,分別解除與共享內存的映射關系。

    后在子進程中刪除共享內存。因為共享內存自身并不提供同步機制,所以應額外實現不同進程間的同步(如信號量)。為了簡單起見,在本實例中用標志字符串來實現非常簡單的父子進程間的同步。

    這里要介紹的一個命令是ipcs,用于報告進程間通信機制狀態,它可以查看共享內存、消息隊列等各種進程間通信機制的情況,這里使用了system()函數調用shell命令“ipcs”。程序源代碼如下:

    /* shmem.c */
    #include <sys/types.h>
    #include <sys/ipc.h>
    #include <sys/shm.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>

    #define BUFFER_SIZE 2048

    int main()
    {
        pid_t pid;
        int shmid;
        char *shm_addr;
        char flag[] = "WROTE";
        char *buff;

        /* 創建共享內存 */
        if ((shmid = shmget(IPC_PRIVATE, BUFFER_SIZE, 0666)) < 0)
        {
            perror("shmget");
            exit(1);
        }
        else
        {
        printf("Create shared-memory: %d\n",shmid);
        }

        /* 顯示共享內存情況 */
        system("ipcs -m");

        pid = fork();
        if (pid == -1)
        {
            perror("fork");
            exit(1);
        }
        else if (pid == 0) /* 子進程處理 */
        {
            /* 映射共享內存 */
            if ((shm_addr = shmat(shmid, 0, 0)) == (void*)-1)
            {
                perror("Child: shmat");
                exit(1);
            }
            else
            {
                printf("Child: Attach shared-memory: %p\n", shm_addr);
            }
            system("ipcs -m");

            /* 通過檢查在共享內存的頭部是否有標志字符串“WROTE”來確認父進程已經向共享內存寫入有效數據 */
            while (strncmp(shm_addr, flag, strlen(flag)))
            {
                printf("Child: Wait for enable data...\n");
                sleep(5);
            }

            /* 獲取共享內存的有效數據并顯示 */
            strcpy(buff, shm_addr + strlen(flag));
            printf("Child: Shared-memory :%s\n", buff);

            /* 解除共享內存映射 */
            if ((shmdt(shm_addr)) < 0)
            {
                perror("shmdt");
                exit(1);
            }
            else
            {
                printf("Child: Deattach shared-memory\n");
            }
            system("ipcs -m");

            /* 刪除共享內存 */
            if (shmctl(shmid, IPC_RMID, NULL) == -1)
            {
                perror("Child: shmctl(IPC_RMID)\n");
                exit(1);
            }
            else
            {
                printf("Delete shared-memory\n");
            }

            system("ipcs -m");
        }
        else /* 父進程處理 */
        {
            /* 映射共享內存 */
            if ((shm_addr = shmat(shmid, 0, 0)) == (void*)-1)
            {
                perror("Parent: shmat");
                exit(1);
            }
            else
            {
                printf("Parent: Attach shared-memory: %p\n", shm_addr);
            }

            sleep(1);
            printf("\nInput some string:\n");
            fgets(buff, BUFFER_SIZE, stdin);
            strncpy(shm_addr + strlen(flag), buff, strlen(buff));
            strncpy(shm_addr, flag, strlen(flag));

            /* 解除共享內存映射 */
            if ((shmdt(shm_addr)) < 0)
            {
                perror("Parent: shmdt");
                exit(1);
            }
            else
            {
                printf("Parent: Deattach shared-memory\n");
            }
            system("ipcs -m");

            waitpid(pid, NULL, 0);
            printf("Finished\n");
        }

        exit(0);
    }

    下面是運行結果,從該結果中可以看出,nattch的值隨著共享內存狀態的變化而變化,共享內存的值根據不同的系統會有所不同。

    $ ./shmem
    Create shared-memory: 753665
    /* 在剛創建共享內存時(尚未有任何地址映射)共享內存的情況 */
    ------ Shared Memory Segments --------
    key        shmid    owner    perms    bytes    nattch    status
    0x00000000 753665   david    666      2048      0

    Child: Attach shared-memory: 0xb7f59000 /* 共享內存的映射地址 */
    Parent: Attach shared-memory: 0xb7f59000
    /* 在父子進程中進行共享內存的地址映射后共享內存的情況 */
    ------ Shared Memory Segments --------
    key        shmid    owner    perms    bytes    nattch    status
    0x00000000 753665   david    666      2048      2

    Child: Wait for enable data...

    Input some string:
    Hello /* 用戶輸入字符串“Hello” */
    Parent: Deattach shared-memory
    /* 在父進程中解除共享內存的映射關系后共享內存的情況 */
    ------ Shared Memory Segments --------
    key        shmid    owner    perms    bytes    nattch    status
    0x00000000 753665   david    666      2048      1
    /* 在子進程中讀取共享內存的有效數據并打印 */
    Child: Shared-memory :hello

    Child: Deattach shared-memory
    /* 在子進程中解除共享內存的映射關系后共享內存的情況 */
    ------ Shared Memory Segments --------
    key        shmid    owner    perms    bytes    nattch    status
    0x00000000 753665   david    666      2048      0

    Delete shared-memory
    /* 在刪除共享內存后共享內存的情況 */
    ------ Shared Memory Segments --------
    key        shmid    owner    perms    bytes    nattch    status

    Finished

    本文選自華清遠見嵌入式培訓教材《從實踐中學嵌入式Linux應用程序開發》

   熱點鏈接:

   1、linux下的信號量
   2、linux下的信號處理實例
   3、信號處理函數signal()和信號集函數組
   4、信號捕捉函數alarm()和pause()
   5、信號發送函數kill()和raise()

更多新聞>> 

主站蜘蛛池模板: 推台锯_多片锯_圆木推台锯_方木多片锯_圆木多片锯-河北茂业机械有限公司 | 油管家,货车油管家,工程车油管家-淄博畅行电子科技有限公司 | 煤炭烘干机|煤泥烘干机|矿泥烘干机|酒糟烘干机|果渣烘干机-东鼎干燥官方网站 | 首页_01精密设备吊装,气垫搬运,半导体设备安装_苏州大方起重吊装公司 | 湖北江南专用特种汽车有限公司官方网站| 可可英语-年轻人的英语听说训练平台 | 自动_链条式_电动推杆_电动开窗器厂家_山东鑫宏玺智能科技有限公司 | 箱式电炉,箱式烧结炉,箱式马弗炉,箱式实验炉,高温箱式电阻炉,箱式加热炉,箱式退火炉,箱式真空气氛炉洛阳研博炉业有限公司 | 柯赛德斯-加美润滑油-专业做汽车工业润滑油品牌加盟代理 | 起点养生网-分享科学养生保健知识,中医健康养生之道-无锡据风网络科技有限公司 | 蒸汽流量计_涡轮流量计_涡街流量计_雷达液位计_污水流量计_分体式_大口径工业流量计-江苏长顺仪表 | 亚洲一区日韩一区欧美一区a,中文字幕乱妇无码AV在线,欧美日韩免费在线观看,国产精品一区二区三区免费,日韩精品免费一线在线观看,日韩一本在线,国产呦精品一区二区三区下载,国产日韩精品一区二区在线观看,欧美日韩高清一区二区三区,日韩在线免费观看视频,欧美日韩一区在线观看 | 上海办公家具_高端实木办公家具_现代智能办公家具定制厂-上海迈亚家具有限公司 | 葫芦岛市鹏翔农药化工科技有限公司-苦参碱可溶液剂-氯氰菊酯乳油-阿维菌素乳油 | 老照片修复,修复老照片,老旧照片翻新-老照片修复网 | 沈阳河山铸造有限公司| 柯赛德斯-加美润滑油-专业做汽车工业润滑油品牌加盟代理 | 深圳市泰美乐纸制品有限公司-纸杯厂,一次性纸杯,广告纸杯,奶茶纸杯,试饮纸杯定做 | 冷却特性测试仪_ 刀柄热胀仪_加油小车价格_皮带张力仪_上海川奇机电设备有限公司 | 南京货架|仓库货架|货架公司|仓储货架工厂批发定做-南京苏正科技实业公司 | 郑州月嫂|月嫂培训|月嫂服务|郑州布卢家政服务有限公司 | 全屋定制超市_全屋定制加盟_星空梵高全屋定制招商 | 天天生活网_实用的生活常识网站、有趣的生活点滴小妙招分享 | 呼吸家官网|肺功能检测仪生产厂家|国产肺功能仪知名品牌|肺功能检测仪|肺功能测试仪|婴幼儿肺功能仪|弥散残气肺功能仪|肺功能测试系统|广州红象医疗科技有限公司|便携式肺功能仪|大肺功能仪|呼吸康复一体机|儿童肺功能仪|肺活量计|医用简易肺功能仪|呼吸康复系统|肺功能仪|弥散肺功能仪(大肺)|便携式肺功能检测仪|肺康复|呼吸肌力测定肺功能仪|肺功能测定仪|呼吸神经肌肉刺激仪|便携式肺功能 | 河北万岁药业有限公司 | 亿企商贸-亿万企业的商务贸易平台-B2B企业产品发布供求信息平台,一带一路中国企业及产品展示平台,免费企业智能自助建站网络营销推广平台,打造B2B企业黄页产品信息发布推广专业综合电子商务平台! | 柱状_椰壳_蜂窝_果壳_粉状活性炭_活性炭厂家 - 銮桦净化 | 小地磅,钢瓶秤,叉车称,轮椅秤,倒桶秤,畜牧秤,轴重仪,称重模块——上海实干实业有限公司-网站首页 | 耐磨涂料_陶瓷涂料_高温涂料_高硬度耐磨涂料-北京耐默科技 | 九江赛璐珞实业有限公司-赛璐珞板,赛璐璐板材,PVC装饰膜,PVC片材,醋酸纤维胶板_多彩装饰材料生产厂家 | 智慧环保大数据云平台_车载走航式扬尘/恶臭/噪声污染源在线监测管控信息系统平台_环保土壤/环境监测系统云平台-奥斯恩智能科技 | 猪粪烘干机|小型鸡粪烘干机|猪粪烘干机价格|小型鸡粪烘干机价格 - 河南宏科重工干燥机设备生产厂家 | 全自动码垛机|码垛机器人|高位码垛机械手|拆垛机|缠绕机|山东腾阳智能装备有限公司 | 在线腐蚀率仪,在线污垢热阻仪,靶式光源仪-北京同德创业科技有限公司 | 河北兴泓环保设备有限公司-冷却塔_废气净化塔_污水除臭设备 | 南尔智能科技 南尔 小南管家 智慧中控屏 智能语音面板 - 深圳市南尔智能科技有限公司 | 汽车漆|汽车油漆|工业油漆涂料|汽车漆加盟-佛山市科涂涂料有限公司 | 上海画册设计-上海宣传册设计-产品手册设计-企业画册设计公司 | 排污管道疏通_长沙消防管道/暗管网漏水检测维修_长沙雨水管道疏通就找湖南鸿磊环保工程有限公司 排水PVC管-PVC排污管-给水PVC管-电线PVC管-米阳建材pvc管厂 | 拓普思(常州)智能科技有限公司-青少年无人机教育培训比赛课程加盟拓活力 | 起名字典|宝宝新生儿男孩女孩起名免费网-诗经周易生辰八字取名字大全 |