[Linux C编程]进程间的通信(二)

时间:2023-01-08 14:52:42

进程间的通信

共享内存

是被多个进程共享的一部分物理内存.共享内存是进程间共享数据的一种最快的方法,一个进程向共享内存区域写入了数据,共享这个内存区域的所有进程就可以立刻看到其中的内容.

 [Linux C编程]进程间的通信(二)

共享内存实现分为两个步骤:

 1、创建共享、打开共享内存,使用shmget函数

 2、映射共享内存,将这段创建的共享内存映射到具体的进程空间去,使用shmat函数

 3. 分离共享内存

 4. 控制、删除共享内存

创建shmget:

函数的作用:在内核中创建共享内存

函数的原型:int shmget(key_t key,int size,int shmflag);

函数的参数:key:键值,IPC_PRIVATE:私有的键值

            size:创建的内存大小

            shmflag:类似open权限

返 回 值:成功共享的内存标识符,出错 -1

头 文 件:#include <sys/ipc.h>

          #include <sys/shm.h>

 

映射shmat:

函数的作用:映射共享内存,映射到各自的内存空间;

函数的原型:void * shmat(int shmat,const void * shmaddr,int flag);

函数的参数:shmid:内存标识符

            shmaddr:映射共享内存到本地进程的指定地址,如果为NULL则由内核自动

                     分配。

            shmflag:SHM_RDONLY:共享内存只读;

                        0:共享内存可读写

分离shmdt:

函数的作用:撤销共享内存的映射

函数的原型:int shmdt(const void * shmaddr);

函数的参数:shmaddr:被映射的共享内存地址

返 回 值:成功:0   出错:-1

 

使用共享内存实现通信:

shm1.c:

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include "shm_com.h"

int main(void)
{
int running=1;
void *shared_memory=(void *)0;
struct shared_use_st *shared_stuff;
int shmid;


/*创建共享内存*/
shmid=shmget((key_t)1234,sizeof(struct shared_use_st),0666|IPC_CREAT);
if(shmid==-1)
{
fprintf(stderr,"shmget failed\n");
exit(EXIT_FAILURE);
}

/*映射共享内存*/
shared_memory=shmat(shmid,(void *)0,0);
if(shared_memory==(void *)-1)
{
fprintf(stderr,"shmat failed\n");
exit(EXIT_FAILURE);
}
printf("Memory attached at %X\n",(int)shared_memory);

/*让结构体指针指向这块共享内存*/
shared_stuff=(struct shared_use_st *)shared_memory;

/*控制读写顺序*/
shared_stuff->written_by_you=0;

/*循环的从共享内存中读数据,直到读到“end”为止*/
while(running)
{
if(shared_stuff->written_by_you)
{
printf("You wrote:%s",shared_stuff->some_text);
sleep(1); //读进程睡一秒,同时会导致写进程睡一秒,这样做到读了之后再写
shared_stuff->written_by_you=0;
if(strncmp(shared_stuff->some_text,"end",3)==0)
{
running=0; //结束循环
}
}
}

/*删除共享内存*/
if(shmdt(shared_memory)==-1)
{
fprintf(stderr,"shmdt failed\n");
exit(EXIT_FAILURE);
}
exit(EXIT_SUCCESS);
}


 

shm2.c:

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include "shm_com.h"

int main(void)
{
int running=1;
void *shared_memory=(void *)0;
struct shared_use_st *shared_stuff;
char buffer[BUFSIZ];
int shmid;

/*创建共享内存*/
shmid=shmget((key_t)1234,sizeof(struct shared_use_st),0666|IPC_CREAT);
if(shmid==-1)
{
fprintf(stderr,"shmget failed\n");
exit(EXIT_FAILURE);
}

/*映射共享内存*/
shared_memory=shmat(shmid,(void *)0,0);
if(shared_memory==(void *)-1)
{
fprintf(stderr,"shmat failed\n");
exit(EXIT_FAILURE);
}
printf("Memory attached at %X\n",(int)shared_memory);

/*让结构体指针指向这块共享内存*/
shared_stuff=(struct shared_use_st *)shared_memory;

/*循环的向共享内存中写数据,直到写入的为“end”为止*/
while(running)
{
while(shared_stuff->written_by_you==1)
{
sleep(1);//等到读进程读完之后再写
printf("waiting for client...\n");
}
printf("Ener some text:");
fgets(buffer,BUFSIZ,stdin);
strncpy(shared_stuff->some_text,buffer,TEXT_SZ);
shared_stuff->written_by_you=1;
if(strncmp(buffer,"end",3)==0)
{
running=0; //结束循环
}
}

/*删除共享内存*/
if(shmdt(shared_memory)==-1)
{
fprintf(stderr,"shmdt failed\n");
exit(EXIT_FAILURE);
}
exit(EXIT_SUCCESS);
}


shm_com.h:

#define TEXT_SZ 2048

struct shared_use_st
{
int written_by_you;
char some_text[TEXT_SZ];
};


 

拓展:使用父子进程实现共同通信。(后续博客会贴出)

 

消息队列:

    消息队列就是一个消息的链表.可以把消息看作一个记录,具有特定的格式.进程可以向中按照一定的规则添加新消息;另一些进程则可以从消息队列中读走消息

 [Linux C编程]进程间的通信(二)

·创建打开消息队列msgget()

·读数据从队列msgrcv()

·写数据到队列msgsnd()

·控制消息队列msgctl()

 

创建msgget:

函数的作用:创建消息队列

函数的原型:int msgget(key_t key,int msgflag);

函数的参数:key 键值:IPC_PRIVATE 设定一个数字,也可以用ftok获得

            msgflag:权限

返 回 值:成功:消息队列ID

          出错:-1

头 文 件:#include <sys/types.h>

          #include <sys/ipc.h>

          #include <sys/shm.h>

 

key_t key;

key = ftok(/home/pipe/magget.c,11);

 

写数据msgsnd:

函数的作用:写数据到消息队列

函数的原型:int msgsnd(int msgid,const void * msggp,size_t msgsize,int msgflag);

函数的参数:msggp 消息结构

                  struct msgbuf

                  {

                       long msgtype; //消息类型

                       char mtext[1];//消息正文

                  };

            msgsize:消息的字节数

            msgflag:IPC_NOWAIT:写不进消息直接返回

                        0:一直等待到能写进去消息为止

返 回 值:成功 0   出错:-1

 

读数据msgrcv:

函数的作用:读数据到消息队列

函数的原型:int magrcv(int msgid,void *msgp,size_t msgsz,long int msgtyp,int msgflg);

函数的参数:msgid:消息队列的队列ID;

            msgp:消息缓冲区,同于msgsnd()函数的msgp;

            msgsz:消息正文的字节数(不包括消息类型指针变量)

            msgtyp:0:接收消息队列中的第一个消息

                    >0:接收消息队列中的第一个类型为msgtyp的消息

                    <0:接收消息队列中的第一个类型值不小于msgtyp绝对值且类型值

                        又最小的消息。

            msgflg:MSG_NOERROR:若返回的消息比msgsz字节多,则消息就会截短到

                   msgsz字节,且不通知消息发送进程。

            IPC_NOWAIT:若在消息队列中并没有相应类型的消息可以接收,则函数立即

                         返回。

             0:msgsnd()调用阻塞直到接收一条相应类型的消息为止。

返 回 值:成功 0,出错 -1

头 文 件:#include <sys/types.h>

          #include <sys/ipc.h>

          #include <sys/shm.h>

 

控制msgctl:

函数的作用:控制消息队列,可以删除消息队列

函数的原型:int msgctl(int msgid,int cmd,struct msgid _ds * buf);

函数的参数:msgid: 消息队列的队列ID

            cmd:IPC_STAT:读取消息队列的结构,存储在buf中

                IPC_SET:设置队列的权限

                IPC_RMID:删除消息队列

返 回 值:成功 0;出错 -1

利用消息队列实现通信:

msg1.c:

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

struct my_msg_st
{
long int my_msg_type;
char some_text[BUFSIZ];
};

int main(void)
{
int running=1;
int msgid;
struct my_msg_st some_data;
long int msg_to_receive=0;

/*创建消息队列*/
msgid=msgget((key_t)1234,0666 | IPC_CREAT);
if(msgid==-1)
{
fprintf(stderr,"msgget failed with error: %d\n",errno);
exit(EXIT_FAILURE);
}
/*循环从消息队列中接收消息*/
while(running)
{
/*读取消息*/
if(msgrcv(msgid,(void *)&some_data,BUFSIZ,msg_to_receive,0)==-1)
{
fprintf(stderr,"msgrcv failed with error: %d\n",errno);
exit(EXIT_FAILURE);
}

printf("You wrote: %s",some_data.some_text);

/*接收到的消息为“end”时结束循环*/
if(strncmp(some_data.some_text,"end",3)==0)
{
running=0;
}
}

/*从系统内核中移走消息队列*/
if(msgctl(msgid,IPC_RMID,0)==-1)
{
fprintf(stderr,"msgctl(IPC_RMID) failed\n");
exit(EXIT_FAILURE);
}
exit(EXIT_SUCCESS);
}


msg2.c:

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#define MAX_TEXT 512

struct my_msg_st
{
long int my_msg_type;
char some_text[MAX_TEXT];
};

int main(void)
{
int running=1;
struct my_msg_st some_data;
int msgid;
char buffer[BUFSIZ];

/*创建消息队列*/
msgid=msgget((key_t)1234,0666 | IPC_CREAT);
if(msgid==-1)
{
fprintf(stderr,"msgget failed with error:%d\n",errno);
exit(EXIT_FAILURE);
}

/*循环向消息队列中添加消息*/
while(running)
{
printf("Enter some text:");
fgets(buffer,BUFSIZ,stdin);
some_data.my_msg_type=1;
strcpy(some_data.some_text,buffer);

/*添加消息*/
if(msgsnd(msgid,(void *)&some_data,MAX_TEXT,0)==-1)
{
fprintf(stderr,"msgsed failed\n");
exit(EXIT_FAILURE);
}

/*用户输入的为“end”时结束循环*/
if(strncmp(buffer,"end",3)==0)
{
running=0;
}
}
exit(EXIT_SUCCESS);
}


信号量:

·信号量(又名:信号灯)与其他进程间通信方式不大相同,主要用途是保护临界资源.

·进程可以根据它判定是否能够访问某些共享资源。除了用于访问控制外,还可用于进程同步

 

1.创建打开信号量,semget()

2.初始化信号量,semctl()的SETVAL操作。当使用二维信号量时将信号量初始化为1;

3. 进行信号量的PV操作,调用semop()函数。

4.如果不用信号量,从系统中删除他/她,使用semctl的IPC_RMID操作。

 

PV操作:

   sem = 1

P  sem - 1 = 0

   sem = 0  hello.c进程A可以使用

V  sem = 1  hello.c进程B可以使用

创建semget:

函数的作用:创建信号量

函数的原型:int semget(key_t key,int nsems, int semflg);

函数的参数:nsems:信号量的数目,通常取1个

            semflag:用open()函数权限位

返 回 值:成功:信号量的标识符

          出错:-1

 

控制 semctl:

函数的作用:信号量的控制:初始化、删除

函数的原型:int semctl(int semid,int semnum,int cmd,union semun,arg);

函数的参数:semnum:通常为0,第一个信号量

            cmd: IPC_STAT:

                IPC_SETVAL:设置为arg中的val的值

                IPC_GETVAL:获取信号量的值

                IPC_RMID:删除信号量

返 回 值:成功:cmd不同返回值也不一样

                   IPC_STAT、IPC_SETVAL、IPC_RMID:返回0

                   IPC_GETVAL:返回为信号量的值

PV操作semop:

函数的作用:执行PV操作

函数的原型:int semop(int semid,struct sembuf *sops, size_t nsops);

 

函数的参数:struct sembuf *sops:

                     sem_op: -1 P操作

                             1:V操作

            nsops:-1

返 回 值:成功:信号量的标识符

          出错:-1

 

 

PV操作举例:

sema.c:

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/sem.h>
#include <sys/ipc.h>
#include <sys/types.h>
#include <sys/stat.h>

int main()
{
int fd = 0;
int ret;
int semid;

struct sembuf sops;

key_t key;

key = ftok("/home/linx",1);

semid = semget(key,1,IPC_CREAT);
ret = semctl(semid,0,SETVAL,1);
printf("Init value of sem is %d.\n",ret);

fd = open("broad.txt",O_RDWR|O_CREAT|O_APPEND);
sops.sem_num = 0;
sops.sem_op = -1;
semop(semid,&sops,1);

write(fd,"class math",11);

sleep(20);
sops.sem_num = 0;
sops.sem_op = +1;
semop(semid,&sops,1);

close(fd);

return 0;
}


semb.c

#include <unistd.h>
#include <stdio.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ipc.h>
#include <sys/sem.h>

int main()
{
int fd = 0;

key_t key;

int semid;

struct sembuf sops;

key = ftok("/home/linux",1);

semid = semget(key,1,IPC_CREAT);

fd = open("./broad.txt",O_RDWR|O_CREAT|O_APPEND);
sops.sem_num = 0;
sops.sem_op = -1;
semop(semid,&sops,1);

write(fd,"english exam",20);
sops.sem_num = 0;
sops.sem_op = +1;
semop(semid,&sops,1);

close(fd);

return 0;
}


执行sema.c semb.c的可执行文件之后,可以cat broad.txt查看结果。