共享内存+互斥量实现linux进程间通信

合集下载

进程间同步的几种方法

进程间同步的几种方法

进程间同步的几种方法进程间同步是指两个或多个进程之间进行协调,以确保它们能够正确地执行。

这是多任务操作系统中的重要问题,因为进程之间共享资源,包括内存、文件和网络连接等。

进程同步的关键是确保一组进程在处理共享资源时,能够避免发生竞态条件(Race Condition)和死锁(Deadlock)。

竞态条件指多个进程同时访问共享资源,导致不正确的结果。

死锁指多个进程互相等待,导致它们都无法继续执行。

1. 互斥锁互斥锁是最常见的同步方法之一,它被用来保护共享资源,确保同一时刻只有一个进程可以访问它。

当一个进程获取了锁,其他进程必须等待,直到锁被释放。

在 POSIX 系统中,互斥锁可以通过 pthread_mutex_t 数据类型实现。

我们可以使用pthread_mutex_init() 函数初始化锁,使用 pthread_mutex_lock() 函数获取锁,使用pthread_mutex_unlock() 函数释放锁。

下面是一个例子,展示了如何使用互斥锁同步两个进程对共享变量的访问:```c#include <pthread.h>#include <stdio.h>int count = 0;pthread_mutex_t lock;void *increment(void *arg) {for (int i = 0; i < 1000000; i++) {pthread_mutex_lock(&lock); // 获取锁count++;pthread_mutex_unlock(&lock); // 释放锁}return NULL;}在上面的例子中,我们创建了两个线程,它们分别对共享变量 count 进行了一百万次的递增操作。

我们使用了互斥锁来保护 count 变量,确保同一时刻只有一个线程可以访问它。

2. 信号量3. 条件变量条件变量可以被用来支持更高级的同步机制,如互斥锁和信号量。

进程间通信的几种方法

进程间通信的几种方法

进程间通信的几种方法进程间通信是计算机系统中一种非常常见的需求,它允许多个进程在不同的地址空间中共享资源,实现信息的共享以及通信。

在计算机系统中,进程间通信的方法会根据使用的网络类型以及网络的连接有所不同。

对于进程间通信的方法,一般可以分为以下几种:(一)共享内存共享内存是一种最简单的进程间通信的方式,也是当今使用最为普遍的进程间通信方法。

在此方法中,多个进程可以访问共享内存区域,这样它们就可以直接在内存中进行通信,而且支持多个进程同时读取和写入内存中的数据,能满足多样化的通信需求,从而提高了系统的效率。

但是,由于这种方法不能实现两个进程之间的“双向”通信,因此它只能适用于一些特定的应用场景,而不能满足一般的进程间通信需求。

(二)消息传递消息传递是进程之间通信的常见方法,它允许两个进程之间进行双向通信,同时还能可靠地传输数据。

在消息传递中,多个进程可以通过将自己的传输内容发送到指定的消息服务器来实现进程间通信。

消息服务器会将这些内容发送到另一个进程,以便双方进行通信。

简单的消息传递本质上是一种客户端/服务器架构,而处理多个进程之间的通信时,可以使用一种名为“发布/订阅”的模型。

在这种模型中,发送者会将消息(即发布)发布到消息服务器上,而接收者(即订阅)可以订阅消息服务器上的那些发布消息。

(三)管道(PIPES)管道是另一种常用的进程间通信模式,它可以实现进程间的双向通信。

在管道模式中,多个进程共享一个双向管道,它们可以在这个双向管道上进行双向通信,也就是说,管道的一端可以用来发送数据,另一端可以用来接收数据。

与消息传递不同,管道不需要使用额外的服务器,因此它可以更快地传输数据,但是它也有很大的局限性,无法跨越网络,仅限于同一台机器上的多个进程之间的通信。

(四)信号量信号量是一种重要的进程间通信机制,它可以用来实现同步和互斥操作,使多个进程都能够按照规定的方式来完成工作,从而实现协作和通信。

信号量原理是通过一个数值来控制多个进程对共享资源的访问,当这个数值为正时,它允许多个进程访问共享资源,当这个数值为0时,它就不允许多个进程访问共享资源。

linux线程间通信方式

linux线程间通信方式

linux线程间通信方式
Linux 线程间通信方式包括以下几种:
1. 管道通信:管道是一种半双工的通信方式,只能用于具有亲缘关系的进程之间的通信,父进程创建管道,在进程间传递数据。

2. 信号通信:信号是一种异步通信方式,在进程之间传递简单的信息。

一个进程向另一个进程发送一个信号,另一个进程收到信号后就可以采取相应的操作。

3. 共享内存通信:共享内存是最快的进程间通信方式,可以将内存区域映射到多个进程的地址空间中,实现进程间数据的共享。

需要注意解决信号量、锁等同步问题。

4. 信号量通信:信号量是一种计数器,用来协调多个进程对共享资源的访问。

多个进程需要对信号量进行操作,以实现对共享资源的访问控制。

5. 消息队列通信:消息队列是一种通过消息传递来进行通信的机制,可以在进程之间传递数据块,通常用于进程间的同步和异步通信。

6. 套接字通信:套接字是基于网络通信的一种进程间通信方式,可用于同一主机上进程间通信,也可以在不同主机之间通信。

套接字是基于 TCP/IP 协议栈实现的,需要在数据传输时注意网络传输和数据结构转换等问题。

以上是 Linux 线程间通信的主要方式,开发者可以根据不同的需求和场景选择合适的方式。

linux下共享内存

linux下共享内存

Linux下共享内存SUNNY.MAN共享内存允许两个或多个进程进程共享同一块内存(这块内存会映射到各个进程自己独立的地址空间) 从而使得这些进程可以相互通信,进程退出时会自动和已经挂接的共享内存区段分离,但是仍建议当进程不再使用共享区段时调用shmdt来卸载区段。

注意,当一个进程分支出父进程和子进程时,父进程先前创建的所有共享内存区段都会被子进程继承。

如果区段已经做了删除标记(在前面以IPC_RMID指令调用shmctl),而当前挂接数已经变为0,这个区段就会被移除。

Linux中通过API函数shmget创建的共享内存一般都是在程序中使用shmctl来释放的,但是有时为了调试程序,开发人员可能通过Ctrl + C等方式发送中断信号来结束程序,此时程序申请的共享内存就不能得到释放,当然如果程序没有改动的话,重新运行程序时仍然会使用上次申请的共享内存,但是如果我们修改了程序,由于共享内存的大小不一致等原因会导致程序申请共享内存错误。

因此,我们总是希望每次结束时就能释放掉申请的共享内存。

有两种方法可以用来释放共享内存:第一种:如果总是通过Crtl+C来结束的话,可以做一个信号处理器,当接收到这个信号的时候,先释放共享内存,然后退出程序。

第二种:不管你以什么方式结束程序,如果共享内存还是得不到释放,那么可以通过linux命令ipcrm shm shmid来释放,在使用该命令之前可以通过ipcs -m命令来查看共享内存。

共享内存查看使用ipcs命令,不加如何参数时,会把共享内存、信号量、消息队列的信息都打印出来,如果只想显示共享内存信息,使用如下命令:[root@localhost ~]# ipcs –m同样共享内存的大小也可以用ipcs –lm来查看它的上限下限。

shmget( ) 创建一个新的共享内存区段取得一个共享内存区段的描述符shmctl( ) 取得一个共享内存区段的信息为一个共享内存区段设置特定的信息移除一个共享内存区段shmat( ) 挂接一个共享内存区段shmdt( ) 于一个共享内存区段的分离同样共享内存的大小也可以用ipcs –lm来查看它的上限下限。

消息队列和共享内存的进程通信机制

消息队列和共享内存的进程通信机制

消息队列和共享内存的进程通信机制
进程间通信是操作系统中非常重要的一部分,因为不同的进程可能需要相互交流信息,共同完成某项任务。

在进程间通信的机制中,消息队列和共享内存是两种常见的方式。

消息队列是一种进程间通信的方式,它是一种先进先出的数据结构,可以用来存储不同进程之间传递的消息。

消息队列有一个消息缓冲区,不同的进程可以向缓冲区中发送消息,并从中接收消息。

发送消息时需要指定消息类型,接收消息时可以选择接收指定类型的消息或者接收所有类型的消息。

共享内存是另一种进程间通信的方式,它是一种可以被多个进程访问的内存区域。

多个进程可以在共享内存中读取和写入数据,这样就可以实现进程间的数据共享和交流。

共享内存的使用需要注意进程同步和互斥的问题,否则可能会导致数据不一致或者错误。

消息队列和共享内存都是进程间通信的有效方式,但是它们各自有自己的优点和缺点。

消息队列适用于进程之间需要传递一些简单的消息,而共享内存适用于需要共享一些复杂的数据结构和大量数据的情况。

选择合适的通信机制可以提高程序的效率和可靠性。

- 1 -。

详解linux进程间通信-消息队列

详解linux进程间通信-消息队列

详解linux进程间通信-消息队列前⾔:前⾯讨论了信号、管道的进程间通信⽅式,接下来将讨论消息队列。

⼀、系统V IPC 三种系统V IPC:消息队列、信号量以及共享内存(共享存储器)之间有很多相似之处。

每个内核中的 I P C结构(消息队列、信号量或共享存储段)都⽤⼀个⾮负整数的标识符( i d e n t i f i e r )加以引⽤。

⽆论何时创建I P C结构(调⽤m s g g e t、 s e m g e t或s h m g e t) ,都应指定⼀个关键字(k e y),关键字的数据类型由系统规定为 k e y _ t,通常在头⽂件< s y s / t y p e s . h >中被规定为长整型。

关键字由内核变换成标识符。

以上简单介绍了IPC,对接下来介绍的消息队列、信号量和共享内存有助于理解。

⼆、消息队列 1、简介 消息队列是消息的链接表 ,存放在内核中并由消息队列标识符标识。

我们将称消息队列为“队列”,其标识符为“队列 I D”。

m s g g e t⽤于创建⼀个新队列或打开⼀个现存的队列。

m s g s n d⽤于将新消息添加到队列尾端。

每个消息包含⼀个正长整型类型字段,⼀个⾮负长度以及实际数据字节(对应于长度),所有这些都在将消息添加到队列时,传送给 m s g s n d。

m s g r c v⽤于从队列中取消息。

我们并不⼀定要以先进先出次序取消息,也可以按消息的类型字段取消息。

2、函数介绍ftok函数#include <sys/types.h>#include <sys/ipc.h>key_t ftok(const char *pathname, int proj_id);//“/home/linux” , 'a'功能:⽣成⼀个key(键值)msgget函数#include <sys/types.h>#include <sys/ipc.h>#include <sys/msg.h>int msgget(key_t key, int msgflg);功能:创建或取得⼀个消息队列对象返回:消息队列对象的id 同⼀个key得到同⼀个对象格式:msgget(key,flag|mode);flag:可以是0或者IPC_CREAT(不存在就创建)mode:同⽂件权限⼀样msgsnd函数int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);功能:将msgp消息写⼊标识为msgid的消息队列msgp:struct msgbuf {long mtype; /* message type, must be > 0 */消息的类型必须>0char mtext[1]; /* message data */长度随意};msgsz:要发送的消息的⼤⼩不包括消息的类型占⽤的4个字节msgflg:如果是0 当消息队列为满 msgsnd会阻塞如果是IPC_NOWAIT 当消息队列为满时不阻塞⽴即返回返回值:成功返回id 失败返回-1msgrcv函数ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp,int msgflg);功能:从标识符为msgid的消息队列⾥接收⼀个指定类型的消息并存储于msgp中读取后把消息从消息队列中删除msgtyp:为 0 表⽰⽆论什么类型都可以接收msgp:存放消息的结构体msgsz:要接收的消息的⼤⼩不包含消息类型占⽤的4字节msgflg:如果是0 标识如果没有指定类型的消息就⼀直等待如果是IPC_NOWAIT 则表⽰不等待msgctl函数int msgctl(int msqid, int cmd, struct msqid_ds *buf);msgctl(msgid,IPC_RMID,NULL);//删除消息队列对象 程序2-2将简单演⽰消息队列: --- snd.c ---#include "my.h"typedef struct{long type;char name[20];int age;}Msg;int main(){key_t key = ftok("/home/liudw",'6');printf("key:%x\n",key);int msgid = msgget(key,IPC_CREAT|O_WRONLY|0777);if(msgid<0){perror("msgget error!");exit(-1);}Msg m;puts("please input your type name age:");scanf("%ld%s%d",&m.type,,&m.age);msgsnd(msgid,&m,sizeof(m)-sizeof(m.type),0);return0;} --- rcv.c ---#include "my.h"typedef struct{long type;char name[20];int age;}Msg;int main(){key_t key = ftok("/home/liudw",'6');printf("key:%x\n",key);int msgid = msgget(key,O_RDONLY);if(msgid<0){perror("msgget error!");exit(-1);}Msg rcv;long type;puts("please input type you want!");scanf("%ld",&type);msgrcv(msgid,&rcv,sizeof(rcv)-sizeof(type),type,0);printf("rcv--name:%s age:%d\n",,rcv.age);msgctl(msgid,IPC_RMID,NULL);return0;} 运⾏演⽰: 三、详解ftok函数 ftok根据路径名,提取⽂件信息,再根据这些⽂件信息及project ID合成key,该路径可以随便设置。

linux sem用法

linux sem用法

linux sem用法Linux系统中的sem(信号量)是一种用于进程间通信和同步的机制。

它提供了一种多个进程之间能够互斥地访问共享资源的方法。

在本文中,我将详细介绍sem的用法,并一步一步地回答您的问题。

第一步:什么是sem?Sem是由Linux内核提供的一种进程同步原语。

它允许进程在对共享资源的访问上进行同步,以避免竞争条件的发生。

它可以用于实现互斥访问共享资源的目的。

第二步:sem的创建和初始化在Linux中,可以使用semget()函数创建一个新的信号量,或者使用semget()函数打开一个已经存在的信号量。

创建新的信号量时,需要指定一个键值,这个键值标识了信号量的唯一性。

创建一个新的信号量的示例代码如下:cint semid = semget(key, 1, IPC_CREAT IPC_EXCL 0666);初始化信号量的值可以使用semctl()函数,其中第二个参数是信号量的编号,第三个参数是命令,第四个参数是信号量的初始值。

初始化信号量的示例代码如下:cunsigned short sem_array[1] = {1}; 初始信号量为1union semun arg;arg.array = sem_array;semctl(semid, 0, SETALL, arg);第三步:sem的P操作和V操作一旦创建和初始化了信号量,就可以使用semop()函数对信号量进行P操作(减法操作)和V操作(加法操作)。

P操作会将信号量的值减1,如果信号量的值为0,则进程将会被阻塞,直到信号量的值大于0为止。

V操作会将信号量的值加1。

如果有多个进程在等待该信号量,则会选择其中一个进行唤醒。

P操作的示例代码如下:cstruct sembuf sb;sb.sem_num = 0;sb.sem_op = -1;sb.sem_flg = SEM_UNDO; semop(semid, &sb, 1);V操作的示例代码如下:cstruct sembuf sb;sb.sem_num = 0;sb.sem_op = 1;sb.sem_flg = SEM_UNDO; semop(semid, &sb, 1);第四步:sem的销毁当信号量不再需要时,可以使用semctl()函数进行销毁。

进程管理实验报告分析(3篇)

进程管理实验报告分析(3篇)

第1篇一、实验背景进程管理是操作系统中的一个重要组成部分,它负责管理计算机系统中所有进程的创建、调度、同步、通信和终止等操作。

为了加深对进程管理的理解,我们进行了一系列实验,以下是对实验的分析和总结。

二、实验目的1. 加深对进程概念的理解,明确进程和程序的区别。

2. 进一步认识并发执行的实质。

3. 分析进程争用资源的现象,学习解决进程互斥的方法。

4. 了解Linux系统中进程通信的基本原理。

三、实验内容1. 使用系统调用fork()创建两个子进程,父进程和子进程分别显示不同的字符。

2. 修改程序,使每个进程循环显示一句话。

3. 使用signal()捕捉键盘中断信号,并通过kill()向子进程发送信号,实现进程的终止。

4. 分析利用软中断通信实现进程同步的机理。

四、实验结果与分析1. 实验一:父进程和子进程分别显示不同的字符在实验一中,我们使用fork()创建了一个父进程和两个子进程。

在父进程中,我们打印了字符'a',而在两个子进程中,我们分别打印了字符'b'和字符'c'。

实验结果显示,父进程和子进程的打印顺序是不确定的,这是因为进程的并发执行。

2. 实验二:每个进程循环显示一句话在实验二中,我们修改了程序,使每个进程循环显示一句话。

实验结果显示,父进程和子进程的打印顺序仍然是随机的。

这是因为并发执行的进程可能会同时占用CPU,导致打印顺序的不确定性。

3. 实验三:使用signal()捕捉键盘中断信号,并通过kill()向子进程发送信号在实验三中,我们使用signal()捕捉键盘中断信号(按c键),然后通过kill()向两个子进程发送信号,实现进程的终止。

实验结果显示,当按下c键时,两个子进程被终止,而父进程继续执行。

这表明signal()和kill()在进程控制方面具有重要作用。

4. 实验四:分析利用软中断通信实现进程同步的机理在实验四中,我们分析了利用软中断通信实现进程同步的机理。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

共享内存+互斥量实现linux进程间通信一、共享内存简介共享内存是进程间通信中高效方便的方式之一。

共享内存允许两个或更多进程访问同一块内存,就如同malloc() 函数向不同进程返回了指向同一个物理内存区域的指针,两个进程可以对一块共享内存进行读写。

共享内存并未提供进程同步机制,使用共享内存完成进程间通信时,需要借助互斥量或者信号量来完成进程的同步。

这里说一下互斥量与信号量的区别。

互斥量用于线程的互斥,信号量用于线程的同步,这是互斥量与信号量的本质区别,其次信号量实现互斥量的功能。

本文结合个人实际项目需求,采用互斥量实现进程间访问共享内存的互斥,即同一时刻只能允许一个进程对共享内存进行写操作。

二、使用系统调用完成共享内存的申请、连接、分离和删除共享内存函数由shmget、shmat、shmdt、shmctl四个函数组成。

使用时需要包含#include<sys/ipc.h>、#include <sys/shm.h>、#include <sys/types.h>和。

1.共享内存的申请使用shmget()完成共享内存的申请,函数原型如下:int shmget(key_t key, size_t size, int shmflg);key:共享内存的标识符,大于0的32位整数。

若是父子关系的进程间通信,这个标识符用IPC_PRIVATE,若进程没有关系,可自行定义。

size:共享内存大小,单位Byte。

shmflg:共享内存的模式(mode),包括三部分,第一部分是:无指定标示符的共享内存是否创建,由0(只获取)或IPC_CREAT(未创建则新建)决定。

第二部分:IPC_EXCL(若已创建,则报错)。

第三部分:权限标识,由八进制表示,如0640,第一个0是八进制数标识,第一个6(4+2)表示拥有者的权限读和写,第二个4表示同组权限写,第3个0表示他人的权限。

这三部分由算数或运算符|拼接组成shmflg,如IPC_CREAT|0640。

成功时返回共享内存的ID,失败时返回-1。

2.共享内存的连接使用shmat()函数将已经申请好的共享连接到当前进程的地址空间,函数原型如下:void *shmat(int shmid, const void *shmaddr, int shmflg);shmid:共享内存标识符。

shmaddr:指定进程使用共享内存的起始地址,直接指定为NULL让内核自己决定一个合适的地址位置。

shmflg:SHM_RDONLY为只读模式,其他为读写模式,通常设置为NULL。

成功时,这个函数返回共享内存的起始地址。

失败时返回-1。

3.共享内存的分离使用sdmdt()函数将已连接的共享内存与进程分离,功能与shmat()相反,函数原型如下:int shmdt(const void *shmaddr);shmaddr:连接的共享内存的起始地址。

成功时返回0。

失败时返回-1。

4.共享内存的删除shmctl() 控制对这块共享内存的使用,包括删除。

函数原型如下:int shmctl(int shmid, int command, struct shmid_ds *buf);shmid:共享内存的ID。

command:是控制命令,IPC_STAT(获取共享内存的状态)、IPC_SET(改变共享内存的状态)IPC_RMID (删除共享内存)。

buf:一个结构体指针。

Command设置为IPC_STAT的时候,取得的状态放在这个结构体中。

如果要改变共享内存的状态,由这个结构体指定。

返回值:成功:0,失败:-1。

三、使用互斥量完成父子进程对共享内存的互斥访问在共享内存中申明互斥量pthread_mutex_t mutex,需要包含<pthread.h>头文件。

使用时的步骤是:第一步:初始化互斥量,使用pthread_mutex_init()函数,函数原型如下:int pthread_mutex_init(pthread_mutex_t *restrict mutex,const pthread_mutexattr_t *restrict attr);使用方法pthread_mutex_init(&mutex,NULL)。

函数成功返回0,失败返回非0。

第二步:对互斥量进程上锁,使用pthread_mutex_lock()函数,函数原型如下:int pthread_mutex_lock(pthread_mutex_t *mutex);函数成功返回0,错误返回非0的错误代码。

第三步:pthread_mutex_unlock()对已经上锁的mutex进行解锁,pthread_mutex_lock()成对出现,函数原型如下:int pthread_mutex_unlock(pthread_mutex_t *mutex)。

函数成功返回0,错误返回非0的错误代码。

第四步:如果最后不需要使用互斥量的话,使用pthread_mutex_destroy()销毁。

函数原型如下:int pthread_mutex_destroy(pthread_mutex_t *mutex);函数成功返回0,错误返回非0的错误代号。

四、示例源码鄙人以实际项目开发过程中实现进程间通信的源码,为大家展示如何利用上面总结的系统调用接口来实现进程间的通信。

1.定义存储成员变量的类,共享内存将用于存放该类的成员数据。

class ALGOriginalTask{public:uint8 algorithm; //算法,在global.h中定义uint8 encodeType; //编码类型0:UTF8,1:UCS2uint32 cipher_num; //密文数目uint32 cipher_len; //密文长度char cipher[128][256]; //密文最多128个uint8 salt_len; //盐的长度char salt[128][32];//最多128个,最长32字节char re[32][256];uint8 plainfile_num;pthread_mutex_t mutex;//读写共享内存互斥量};class ALGResult{public:bool init();ALGResult() {};~ALGResult() {};struct timeval ts;struct timeval te;uint8 success_num;uint8 algorithm;uint8 cipher_len;char cipher[128][256];char salt[128][32];char ret_plain[MAX_CIPHER_NUM][MAX_PLAIN_LEN];};2.使用fork()在父进程中产生子进程,通过系统调用execve()调用另外可执行程序ALG.out,将ALGOriginalTask中的数据写入共享内存后,再在ALG.out中通过接收到的shmid获取共享内存进行读取。

最后在ALG.out中将ALGResult中的数据写入共享内存,再由父进程读取结果。

父进程代码:int main(int argc, char **argv){pid_t pid;//子进程IDint shmid;//共享内存IDALGOriginalTask* ALG_shm_addr=NULL;ALGResult* ALG_Result_shm_addr=NULL;void* shm_addr=NULL;//共享内存首地址//char buff[sizeof(ALGOriginalTask)];if((shmid=shmget(IPC_PRIVATE,sizeof(ALGOriginalTask)+sizeof(ALGResult),IPC_CREAT|0640))<0)//创建当前进程的父子进程私有共享内存{perror("shmget");exit(-1);} else{//create successfullyprintf("Create shared memory: %d.\n",shmid);printf("Created shared memory status:\n");system("ipcs -m");}printf("start fork\n");switch(pid=fork()){case -1:perror("fork");exit(-1);case 0://child{//构造ALG.out参数string shmId=itoa(shmid);string plain_path=ALG_task->plainfile_path;string plain_offset=itoa(ALG_task->plain_offset);string plain_num=itoa(ALG_task->plain_num);string plain_len=itoa(ALG_task->plain_len);cout<<"start execute ALG.out"<<endl;const char* argv[10]={"./ALG.out",shmId.c_str(),plain_path.c_str(),plain_offset.c_str(),plain_num.c_str( ),plain_len.c_str()};execlp(argv[0], argv[0], argv[1], argv[2], argv[3],argv[4], argv[5], NULL);exit(0);}default://parent{if((shm_addr=shmat(shmid,0,0))==(void*)-1){perror("Parent:shmat");//打印函数执行错误原因,Parent:shmat:错误原因exit(-1);}else{//初始化共享内存memset(shm_addr,NULL,sizeof(ALGOriginalTask)+sizeof(ALGResult));printf("Parent: Attach shared-memory: %p\n",shm_addr);printf("Parent Attach shared memory status:\n");system("ipcs -m");}//设置共享内存格式ALG_shm_addr=static_cast<ALGOriginalTask*>(shm_addr);//向共享内存写入参数pthread_mutex_init(&ALG_shm_addr->mutex,NULL);pthread_mutex_lock(&ALG_shm_addr->mutex);//初始化originalTaskcout<<"writeALG_original_task->algorithm:"<<(int)ALG_original_task->algorithm<<endl;ALG_shm_addr->algorithm=ALG_original_task->algorithm;ALG_shm_addr->encodeType=ALG_original_task->encodeType;ALG_shm_addr->cipher_num=ALG_original_task->cipher_num;ALG_shm_addr->cipher_len=ALG_original_task->cipher_len;ALG_shm_addr->salt_len=ALG_original_task->salt_len;//memcpy(ALG_shm_addr->task_name,ALG_original_task->task_name,20);memcpy(ALG_shm_addr->cipher,ALG_original_task->cipher,128*256);memcpy(ALG_shm_addr->salt,ALG_original_task->salt,128*32);memcpy(ALG_shm_addr->re,ALG_original_task->re,32*256);pthread_mutex_unlock(&ALG_shm_addr->mutex);waitpid(pid,NULL,0);//等待子进程计算结束,从共享内存读取返回结果ALG_Result_shm_addr=static_cast<ALGResult*>(shm_addr);//设置共享内存格式//读取共享内存数据,即计算结果memcpy(&ALG_Result->ts,&ALG_Result_shm_addr->ts,sizeof(struct timeval));memcpy(&ALG_Result->te,&ALG_Result_shm_addr->te,sizeof(struct timeval));ALG_Result->success_num=ALG_Result_shm_addr->success_num;memcpy(ALG_Result->ret_plain,ALG_Result_shm_addr->ret_plain,128*64);//删除父进程的共享内存映射地址if (shmdt(shm_addr)<0) {perror("Parent:shmdt");exit(1);}elseprintf("Parent: Deattach shared-memory.\n");//删除共享内存if (shmctl(shmid,IPC_RMID,NULL)==-1){perror("shmct:IPC_RMID");exit(-1);}elseprintf("Delete shared-memory.\n");}}//end forkALG_Result->algorithm=ALG_original_task->algorithm;memcpy(ALG_Result->cipher,ALG_original_task->cipher,128*256);memcpy(ALG_Result->salt,ALG_original_task->salt,128*32);cout<<"computation end"<<endl;//succeedreturn ALG_Result;}ALG.out代码:int main(int argc, char **argv){gettimeofday(&tt1,NULL);ALGOriginalTask* ALG_shm_addr=NULL;ALGResult* ALG_Result_shm_addr=NULL;GPR_Info* gpr_Info=new GPR_Info;RST_Info* rst_Info=new RST_Info;memset(gpr_Info,NULL,sizeof(GPR_Info));memset(rst_Info,NULL,sizeof(RST_Info));void* shm_addr=NULL;//共享内存首地址int shmid=atoi(argv[1]);if((shm_addr=shmat(shmid,0,0))==(void*)-1){perror("Child_ALG.out:shmat");exit(-1);}//设置共享内存格式ALG_shm_addr=static_cast<ALGOriginalTask*>(shm_addr);//waiting for the parent write data to share memorycout<<"wait parent write sharememory:"<<endl;while(1){if((int)ALG_shm_addr->algorithm>=0)break;}cout<<"start read gpr_Info from sharememory:"<<endl;//get info of GPRpthread_mutex_lock(&ALG_shm_addr->mutex);//lock waiting for parent process finish the writing of share memorygpr_Info->algorithm=ALG_shm_addr->algorithm;gpr_Info->encodeType=ALG_shm_addr->encodeType;gpr_Info->cipher_num=ALG_shm_addr->cipher_num;gpr_Info->cipher_len=ALG_shm_addr->cipher_len;gpr_Info->plain_len=atoi(argv[5]);gpr_Info->salt_len=ALG_shm_addr->salt_len;memcpy(gpr_Info->salt,ALG_shm_addr->salt,128*32);memcpy(gpr_Info->cipher,ALG_shm_addr->cipher,128*256);pthread_mutex_unlock(&ALG_shm_addr->mutex);//unlock the mutex//获取计算结果get_ret_palin(ret_plain, gpr_Info);//write result to share memorymemset(shm_addr,NULL,sizeof(ALGOriginalTask)+sizeof(ALGResult));ALG_Result_shm_addr=static_cast<ALGResult*>(shm_addr);ALG_Result_shm_addr->ts=tt1;ALG_Result_shm_addr->te=tt2;ALG_Result_shm_addr->algorithm=gpr_Info->algorithm;ALG_Result_shm_addr->cipher_len=gpr_Info->cipher_len;ALG_Result_shm_addr->success_num=rst_Info->success_num;memcpy(ALG_Result_shm_addr->salt,gpr_Info->salt,128*32);memcpy(ALG_Result_shm_addr->cipher,gpr_Info->cipher,128*256);memcpy(ALG_Result_shm_addr->ret_plain,ret_plain,128*64);//删除子进程的共享内存映射地址if (shmdt(shm_addr)<0){perror("ALG.out:shmdt");exit(1);}elseprintf("ALG.out:Detach shared-memory.\n");delete []charset;delete []ret_plain;return 1;}参考文献:[1]百度百科.共享内存[2] /ljianhui/article/details/10253345。

相关文档
最新文档