ToB企服应用市场:ToB评测及商务社交产业平台

标题: 进程间通信总结 [打印本页]

作者: 罪恶克星    时间: 2023-6-27 09:15
标题: 进程间通信总结
2023/6/27  通信之间的实现,目的是为了,方便回顾时立马能使用。具体的比如什么情况选用什么通信,各自的优点,没有记录。原因是,本人目前实战经验较少,还处于学习状态,对此的理解还停留在管道方便,共享内存效率高,控制操作用消息队列
前提

答:7种,分别为无名管道、有名管道、信号、消息队列、共享内存、信号灯集、套接字
答:
答:如下所示
无名管道

函数接口
  1. int pipe(int fd[2])
  2. 功能:创建无名管道
  3. 参数:文件描述符 fd[0]:读端  fd[1]:写端
  4. 返回值:成功 0
  5.       失败 -1
复制代码
注意事项

a.  当管道中无数据时,读操作会阻塞;
管道中无数据时,将写端关闭,读操作会立即返回
b. 管道中装满(管道大小64K)数据写阻塞,一旦有4k空间,写继续
c. 只有在管道的读端存在时,向管道中写入数据才有意义。否则,会导致管道破裂,向管道中写入数据的进程将收到内核传来的SIGPIPE信号  (通常Broken pipe错误)。
功能实现

父进程循环从终端输入字符串,子进程将字符串循环输出
  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <unistd.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include <sys/wait.h>
  7. int main(int argc, char const *argv[])
  8. {
  9.     char buf[32] = "";
  10.     pid_t id;
  11.     int fd[2] = {0};
  12.     if(pipe(fd) < 0)
  13.     {
  14.         perror("pipe err");
  15.         return -1;
  16.     }
  17.     if ((id = fork()) < 0)
  18.     {
  19.         perror("fork err");
  20.         return -1;
  21.     }
  22.     else if (id == 0)//子进程
  23.     {
  24.         while(1)
  25.         {
  26.             char buf[32] = "";
  27.             read(fd[0], buf, 32);
  28.             if(strcmp(buf, "quit") == 0)
  29.                 break;
  30.             printf("buf:%s\n", buf);
  31.         }
  32.         exit(0);
  33.     }
  34.     else//父进程
  35.     {
  36.         while(1)
  37.         {
  38.             char buf[32] = "";
  39.             // scanf("%s", buf);
  40.             fgets(buf, 32, stdin);
  41.             write(fd[1], buf, strlen(buf)+1);
  42.             if(strcmp(buf, "quit") == 0)
  43.                 break;
  44.         }
  45.         wait(NULL);
  46.     }
  47.     return 0;
  48. }
复制代码
有名管道

函数接口
  1. int mkfifo(const char *filename,mode_t mode);
  2. 功能:创健有名管道
  3. 参数:filename:有名管道文件名
  4.        mode:权限
  5. 返回值:成功:0
  6.        失败:-1,并设置errno号
  7. 注意对错误的处理方式:
  8. 如果错误是file exist时,注意加判断,如:if(errno == EEXIST)
复制代码
注意事项

a. 只写方式,写阻塞,一直到另一个进程把读打开
b. 只读方式,读阻塞,一直到另一个进程把写打开
c. 可读可写,如果管道中没有数据,读阻塞
功能实现

读写操作
  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <sys/stat.h>
  4. #include <errno.h>
  5. #include <unistd.h>
  6. #include <string.h>
  7. #include <fcntl.h>
  8. int main(int argc, char const *argv[])
  9. {
  10.     int fd;
  11.     //创建管道
  12.     if (mkfifo("./fifo", 0666) < 0)
  13.     {
  14.         if (errno == EEXIST)
  15.             printf("file exists\n");
  16.         else
  17.         {
  18.             perror("mkfifo err");
  19.             return -1;
  20.         }
  21.     }
  22.     printf("mkfifo ok\n");
  23.     //打开管道
  24.     fd = open("./fifo", O_RDWR);
  25.     if(fd < 0)
  26.     {
  27.         perror("open err");
  28.         return -1;
  29.     }
  30.     char buf[32] = "hello";
  31.     char data[32] = "";
  32.     write(fd, buf, strlen(buf));
  33.     read(fd, data, 32);
  34.     printf("%s\n", data);
  35.     return 0;
  36. }
复制代码
信号

函数接口
  1. int kill(pid_t pid, int sig);
  2. 功能:信号发送
  3. 参数:pid:指定进程
  4.    sig:要发送的信号
  5. 返回值:成功 0     
  6.        失败 -1
  7. int raise(int sig);
  8. 功能:进程向自己发送信号
  9. 参数:sig:信号
  10. 返回值:成功 0   
  11.        失败 -1
  12. signal(SIGINT, SIG_IGN);//忽略信号
  13. signal(SIGINT, SIG_DFL); //执行默认操作
  14. signal(SIGINT, handler); //捕捉信号
  15. void handler(int sig)//捕捉到该信号执行该函数操作
  16. {
  17.     printf("ctrl+c\n");
  18. }
复制代码
注意事项

SIGKILL:结束进程,不能被忽略不能被捕捉
SIGSTOP:结束进程,不能被忽略不能被捕捉
SIGCHLD:子进程状态改变时给父进程发的信号,不会结束进程
SIGINT:结束进程,对应快捷方式ctrl+c
SIGTSTP:暂停信号,对应快捷方式ctrl+z
SIGQUIT:退出信号,对应快捷方式ctrl+
SIGALRM:闹钟信号,alarm函数设置定时,当到设定的时间时,内核会向进程发送此信号结束进程。
SIGTERM:结束终端进程,kill 使用时不加数字默认是此信号
功能实现

用信号的知识实现司机和售票员问题。
1)售票员捕捉SIGINT(代表开车)信号,向司机发送SIGUSR1信号,司机打印(let's gogogo)
2)售票员捕捉SIGQUIT(代表停车)信号,向司机发送SIGUSR2信号,司机打印(stop the bus)
3)司机捕捉SIGTSTP(代表到达终点站)信号,向售票员发送SIGUSR1信号,售票员打印(please get off the bus)
4)司机等待售票员下车,之后司机再下车。
  1. #include <stdio.h>
  2. #include <signal.h>
  3. #include <unistd.h>
  4. #include <stdlib.h>
  5. #include <sys/types.h>
  6. #include <sys/wait.h>
  7. pid_t pid;
  8. void driver(int sig)
  9. {
  10.     if (sig == SIGUSR1)
  11.         printf("let's gogogo\n");
  12.     else if (sig == SIGUSR2)
  13.         printf("stop the bus\n");
  14.     else if (sig == SIGTSTP)
  15.     {
  16.         kill(pid, SIGUSR1);
  17.         wait(NULL);
  18.         exit(0);
  19.     }
  20. }
  21. void saler(int sig)
  22. {
  23.     if (sig == SIGINT)
  24.         kill(getppid(), SIGUSR1);
  25.     else if (sig == SIGQUIT)
  26.         kill(getppid(), SIGUSR2);
  27.     else if (sig == SIGUSR1)
  28.     {
  29.         printf("please get off the bus\n");
  30.         exit(0);
  31.     }
  32. }
  33. int main(int argc, char const *argv[])
  34. {
  35.     if ((pid = fork()) < 0)
  36.     {
  37.         perror("fork err");
  38.         return -1;
  39.     }
  40.     else if (pid == 0)
  41.     {
  42.         signal(SIGINT, saler);
  43.         signal(SIGQUIT, saler);
  44.         signal(SIGUSR1, saler);
  45.         signal(SIGTSTP, SIG_IGN);
  46.     }
  47.     else
  48.     {
  49.         signal(SIGUSR1, driver);
  50.         signal(SIGUSR2, driver);
  51.         signal(SIGTSTP, driver);
  52.         signal(SIGINT, SIG_IGN);
  53.         signal(SIGQUIT, SIG_IGN);
  54.     }
  55.     while (1)
  56.         pause();
  57.     return 0;
  58. }
复制代码
消息队列

函数接口
  1. 步骤:
  2. 1)创建key值  ftok
  3. 2)创建或打开消息队列  msgget  
  4. 3)添加消息/读取消息   msgsnd/msgrcv
  5. 4)删除消息队列  msgctl
  6. 命令:
  7. ipcs  -q :查看消息队列
  8. ipcrm  -q  msgid :删除消息队列
  9. int msgget(key_t key, int flag);
  10.         功能:创建或打开一个消息队列
  11.         参数:  key值
  12.        flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666
  13.         返回值:成功:msgid
  14.        失败:-1
  15. int msgsnd(int msqid, const void *msgp, size_t size, int flag);
  16.                 功能:添加消息
  17.                 参数:msqid:消息队列的ID
  18.       msgp:指向消息的指针。常用消息结构msgbuf如下:
  19.           struct msgbuf{
  20.             long mtype;        //消息类型
  21.             char mtext[N]};   //消息正文
  22.    size:发送的消息正文的字节数
  23.    flag:IPC_NOWAIT消息没有发送完成函数也会立即返回   
  24.          0:直到发送完成函数才返回
  25.                 返回值:成功:0
  26.       失败:-1
  27.                 使用:msgsnd(msgid, &msg,sizeof(msg)-sizeof(long), 0)
  28.                 注意:消息结构除了第一个成员必须为long类型外,其他成员可以根据应用的需求自行定义。
  29. int msgrcv(int msgid,  void* msgp,  size_t  size,  long msgtype,  int  flag);
  30.         功能:读取消息
  31.         参数:msgid:消息队列的ID
  32.      msgp:存放读取消息的空间
  33.      size:接受的消息正文的字节数
  34.     msgtype:0:接收消息队列中第一个消息。
  35.             大于0:接收消息队列中第一个类型为msgtyp的消息.
  36.             小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。
  37.      flag:0:若无消息函数会一直阻塞
  38.         IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG
  39.                 返回值:成功:接收到的消息的长度
  40.       失败:-1
  41. int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
  42.                 功能:对消息队列的操作,删除消息队列
  43.                 参数:msqid:消息队列的队列ID
  44.      cmd:
  45.         IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。
  46.         IPC_SET:设置消息队列的属性。这个值取自buf参数。
  47.         IPC_RMID:从系统中删除消息队列。
  48.      buf:消息队列缓冲区
  49.         返回值:成功:0
  50.       失败:-1
  51.                 用法:msgctl(msgid, IPC_RMID, NULL)
复制代码
功能代码

基础实现
  1. struct msgbuf{
  2.     long type;
  3.     int num;
  4.     char buf[32];
  5. };
  6. int main(int argc, char const *argv[])
  7. {
  8.     key_t key;
  9.     int msgid;
  10.     //创建key值
  11.     key = ftok("./app", 'b');
  12.     if (key < 0)
  13.     {
  14.         perror("ftok err");
  15.         return -1;
  16.     }
  17.     printf("%#x\n", key);
  18.     //创建消息队列
  19.     msgid = msgget(key, IPC_CREAT|IPC_EXCL|0666);
  20.     if(msgid < 0)
  21.     {
  22.         if(errno == EEXIST)
  23.             msgid = msgget(key, 0666);
  24.         else
  25.         {
  26.             perror("msgget err");
  27.             return -1;
  28.         }     
  29.     }
  30.     //添加消息
  31.     int size = sizeof(struct msgbuf)-sizeof(long);
  32.     struct msgbuf msg = {1, 100, "hello"};
  33.     struct msgbuf msg1 = {2, 200, "world"};
  34.     msgsnd(msgid, &msg, size, 0);
  35.     msgsnd(msgid, &msg1, size, 0);
  36.     //读取消息
  37.     struct msgbuf m;
  38.     msgrcv(msgid, &m, size, 2, 0);
  39.     printf("%d %s\n", m.num, m.buf);
  40.     //删除消息队列
  41.     msgctl(msgid, IPC_RMID, NULL);
  42.     return 0;
  43. }
复制代码
两个进程通过消息队列进行通信,一个进程从终端输入下发的指令,另一个进程接收指令,并打印对应操作语句。
如果输入LED ON,另一个进程输出 “开灯”
如果输入LED OFF,另一个进程输出 “关灯”
  1. //send端
  2. #include <stdio.h>
  3. #include <stdio.h>
  4. #include <sys/types.h>
  5. #include <sys/ipc.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. #include <sys/msg.h>
  9. struct msgbuf
  10. {
  11.     long type;
  12.     char buf[32];
  13. };
  14. int main(int argc, char const *argv[])
  15. {
  16.     key_t key;
  17.     int msgid;
  18.     //创建key值
  19.     key = ftok("./app", 'b');
  20.     if (key < 0)
  21.     {
  22.         perror("ftok err");
  23.         return -1;
  24.     }
  25.     printf("%#x\n", key);
  26.     //创建消息队列
  27.     msgid = msgget(key, IPC_CREAT|IPC_EXCL|0666);
  28.     if(msgid < 0)
  29.     {
  30.         if(errno == EEXIST)
  31.             msgid = msgget(key, 0666);
  32.         else
  33.         {
  34.             perror("msgget err");
  35.             return -1;
  36.         }     
  37.     }
  38.     struct msgbuf msg;
  39.     msg.type = 1;
  40.     int s = sizeof(struct msgbuf)-sizeof(long);
  41.     while(1)
  42.     {
  43.         scanf("%s", msg.buf);
  44.         msgsnd(msgid, &msg, s, 0);
  45.     }
  46.     return 0;
  47. }
复制代码
  1. //recv端
  2. #include <stdio.h>
  3. #include <stdio.h>
  4. #include <sys/types.h>
  5. #include <sys/ipc.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. #include <sys/msg.h>
  9. struct msgbuf
  10. {
  11.     long type;
  12.     char buf[32];
  13. };
  14. int main(int argc, char const *argv[])
  15. {
  16.     key_t key;
  17.     int msgid;
  18.     //创建key值
  19.     key = ftok("./app", 'b');
  20.     if (key < 0)
  21.     {
  22.         perror("ftok err");
  23.         return -1;
  24.     }
  25.     printf("%#x\n", key);
  26.     //创建消息队列
  27.     msgid = msgget(key, IPC_CREAT|IPC_EXCL|0666);
  28.     if(msgid < 0)
  29.     {
  30.         if(errno == EEXIST)
  31.             msgid = msgget(key, 0666);
  32.         else
  33.         {
  34.             perror("msgget err");
  35.             return -1;
  36.         }     
  37.     }
  38.     struct msgbuf msg;
  39.     int s = sizeof(struct msgbuf)-sizeof(long);
  40.     while(1)
  41.     {
  42.         msgrcv(msgid, &msg, s, 1, 0);
  43.         if(strcmp(msg.buf, "LEDON") == 0)
  44.             printf("开灯\n");
  45.         else if(strcmp(msg.buf, "LEDOFF") == 0)
  46.             printf("关灯\n");
  47.     }
  48.    
  49.     return 0;
  50. }
复制代码
共享内存

函数接口
  1. //步骤
  2. 0)创建key值       
  3. 1)创建或打开共享内存
  4. 2)映射
  5. 3)取消映射
  6. 4)删除共享内存
  7. key_t ftok(const char *pathname, int proj_id);
  8. 功能:创建key值
  9. 参数:pathname:文件名
  10.      proj_id:取整型数的低8位数值
  11. 返回值:成功:key值
  12.        失败:-1
  13. int shmget(key_t key, size_t size, int shmflg);
  14. 功能:创建或打开共享内存
  15. 参数:
  16.     key  键值
  17.     size   共享内存的大小
  18.     shmflg   IPC_CREAT|IPC_EXCL|0777
  19. 返回值:成功   shmid
  20.       出错    -1
  21. void  *shmat(int  shmid,const  void  *shmaddr,int  shmflg);
  22. 功能:映射共享内存,即把指定的共享内存映射到进程的地址空间用于访问
  23. 参数:
  24.     shmid   共享内存的id号
  25.     shmaddr   一般为NULL,表示由系统自动完成映射
  26.               如果不为NULL,那么有用户指定
  27.     shmflg:SHM_RDONLY就是对该共享内存只进行读操作
  28.                 0     可读可写
  29. 返回值:成功:完成映射后的地址,
  30.       出错:-1的地址
  31. 用法:if((p = (char *)shmat(shmid,NULL,0)) == (char *)-1)
  32. int shmdt(const void *shmaddr);
  33. 功能:取消映射
  34. 参数:要取消的地址
  35. 返回值:成功0  
  36.       失败的-1
  37. int  shmctl(int  shmid,int  cmd,struct  shmid_ds   *buf);
  38. 功能:(删除共享内存),对共享内存进行各种操作
  39. 参数:
  40.     shmid   共享内存的id号
  41.     cmd     IPC_STAT 获得shmid属性信息,存放在第三参数
  42.             IPC_SET 设置shmid属性信息,要设置的属性放在第三参数
  43.             IPC_RMID:删除共享内存,此时第三个参数为NULL即可
  44. 返回:成功0
  45.      失败-1
  46. 用法:shmctl(shmid,IPC_RMID,NULL);
复制代码
功能代码

基础
  1. int main(int argc, char const *argv[])
  2. {
  3.     key_t key;
  4.     int shmid;
  5.     //创建key值
  6.     key = ftok("./app", 'b');
  7.     if (key < 0)
  8.     {
  9.         perror("ftok err");
  10.         return -1;
  11.     }
  12.     printf("%#x\n", key);
  13.     //创建或打印共享内存
  14.     shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666);
  15.     if (shmid < 0)
  16.     {
  17.         if (errno == EEXIST)
  18.             shmid = shmget(key, 128, 0666);
  19.         else
  20.         {
  21.             perror("shmget err");
  22.             return -1;
  23.         }
  24.     }
  25.     printf("shmid:%d\n", shmid);
  26.     //映射
  27.     char *p = NULL;
  28.     p = shmat(shmid, NULL, 0); //NULL:系统自动进行映射 0:可读可写
  29.     if(p == (char *)-1)
  30.     {
  31.         perror("shmat err");
  32.         return -1;
  33.     }
  34.     strcpy(p, "hello");
  35.     printf("%s\n", p);
  36.     //取消映射
  37.     shmdt(p);
  38.     //删除共享内存
  39.     shmctl(shmid, IPC_RMID, NULL);
  40.     return 0;
  41. }
复制代码
练习:一个进程从终端输入,另一个进程将数据输出,借助共享内存通信。
要求:当输入quit时程序退出
同步:标志位
  1. //read.c
  2. #include <stdio.h>
  3. #include <sys/types.h>
  4. #include <sys/ipc.h>
  5. #include <sys/shm.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. struct msg
  9. {
  10.     int flg;
  11.     char buf[32];
  12. };
  13. int main(int argc, char const *argv[])
  14. {
  15.     key_t key;
  16.     int shmid;
  17.     //创建key值
  18.     key = ftok("./app", 'b');
  19.     if (key < 0)
  20.     {
  21.         perror("ftok err");
  22.         return -1;
  23.     }
  24.     printf("%#x\n", key);
  25.     //创建或打印共享内存
  26.     shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666);
  27.     if (shmid < 0)
  28.     {
  29.         if (errno == EEXIST)
  30.             shmid = shmget(key, 128, 0666);
  31.         else
  32.         {
  33.             perror("shmget err");
  34.             return -1;
  35.         }
  36.     }
  37.     printf("shmid:%d\n", shmid);
  38.     //映射
  39.     struct msg *p = NULL;
  40.     p = (struct msg *)shmat(shmid, NULL, 0); //NULL:系统自动进行映射 0:可读可写
  41.     if(p == (struct msg *)-1)
  42.     {
  43.         perror("shmat err");
  44.         return -1;
  45.     }
  46.     p->flg = 0;
  47.     while(1)
  48.     {
  49.         scanf("%s", p->buf);
  50.         p->flg = 1;
  51.         if(strcmp(p->buf, "quit") == 0)
  52.             break;
  53.     }
  54.     return 0;
  55. }
复制代码
  1. //write.c
  2. #include <stdio.h>
  3. #include <sys/types.h>
  4. #include <sys/ipc.h>
  5. #include <sys/shm.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. struct msg
  9. {
  10.     int flg;
  11.     char buf[32];
  12. };
  13. int main(int argc, char const *argv[])
  14. {
  15.     key_t key;
  16.     int shmid;
  17.     //创建key值
  18.     key = ftok("./app", 'b');
  19.     if (key < 0)
  20.     {
  21.         perror("ftok err");
  22.         return -1;
  23.     }
  24.     printf("%#x\n", key);
  25.     //创建或打印共享内存
  26.     shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666);
  27.     if (shmid < 0)
  28.     {
  29.         if (errno == EEXIST)
  30.             shmid = shmget(key, 128, 0666);
  31.         else
  32.         {
  33.             perror("shmget err");
  34.             return -1;
  35.         }
  36.     }
  37.     printf("shmid:%d\n", shmid);
  38.     //映射
  39.     struct msg *p = NULL;
  40.     p = shmat(shmid, NULL, 0); //NULL:系统自动进行映射 0:可读可写
  41.     if (p == (struct msg *)-1)
  42.     {
  43.         perror("shmat err");
  44.         return -1;
  45.     }
  46.     p->flg = 0;
  47.     while (1)
  48.     {
  49.         if (p->flg == 1)
  50.         {
  51.             if (strcmp(p->buf, "quit") == 0)
  52.                 break;
  53.             printf("data:%s\n", p->buf);
  54.             p->flg = 0;
  55.         }
  56.     }
  57.     //取消映射
  58.     shmdt(p);
  59.     //删除共享内存
  60.     shmctl(shmid, IPC_RMID, NULL);
  61.     return 0;
  62. }
复制代码
信号灯集

函数接口
  1. 步骤
  2.         0)创建key值
  3.         1)创建或打开信号灯集   semget
  4.         2)初始化信号灯集  semctl
  5.         3)pv操作  semop
  6.         4)删除信号灯集  semctl
  7. int semget(key_t key, int nsems, int semflg);
  8. 功能:创建/打开信号灯
  9. 参数:key:ftok产生的key值
  10.     nsems:信号灯集中包含的信号灯数目
  11.     semflg:信号灯集的访问权限,通常为IPC_CREAT |0666
  12. 返回值:成功:信号灯集ID
  13.        失败:-1
  14. int semop ( int semid, struct sembuf  *opsptr,  size_t  nops);
  15. 功能:对信号灯集合中的信号量进行PV操作
  16. 参数:semid:信号灯集ID
  17.      opsptr:操作方式
  18.      nops:  要操作的信号灯的个数 1个
  19. 返回值:成功 :0
  20.       失败:-1
  21. struct sembuf {
  22.    short  
  23.        ; // 要操作的信号灯的编号
  24.    short  sem_op;  //    0 :  等待,直到信号灯的值变成0
  25.                    //   1  :  释放资源,V操作
  26.                    //   -1 :  分配资源,P操作                    
  27.     short  sem_flg; // 0(阻塞),IPC_NOWAIT, SEM_UNDO
  28. };
  29. 用法:
  30. 申请资源 P操作:
  31.     mysembuf.sem_num = 0;
  32.     mysembuf.sem_op = -1;
  33.     mysembuf.sem_flg = 0;
  34.     semop(semid, &mysembuf, 1);
  35. 释放资源 V操作:
  36.     mysembuf.sem_num = 0;
  37.     mysembuf.sem_op = 1;
  38.     mysembuf.sem_flg = 0;
  39.     semop(semid, &mysembuf, 1);
  40. int semctl ( int semid, int semnum,  int cmd…/*union semun arg*/);
  41. 功能:信号灯集合的控制(初始化/删除)
  42. 参数:semid:信号灯集ID
  43.     semnum: 要操作的集合中的信号灯编号
  44.      cmd:
  45.         GETVAL:获取信号灯的值,返回值是获得值
  46.         SETVAL:设置信号灯的值,需要用到第四个参数:共用体
  47.         IPC_RMID:从系统中删除信号灯集合
  48. 返回值:成功 0
  49.       失败 -1
  50. 用法:初始化:
  51. union semun{
  52.     int val; //信号灯的初值
  53. }mysemun;
  54. mysemun.val = 10;
  55. semctl(semid, 0, SETVAL, mysemun);
  56. 获取信号灯值:函数semctl(semid, 0, GETVAL)的返回值
  57. 删除信号灯集:semctl(semid, 0, IPC_RMID);
  58. ipcs -s:查看信号灯集
  59. ipcrm -s  semid:删除信号灯集
复制代码
功能实现
  1. union semun {
  2.     int val; //信号灯的初值
  3. };
  4. int main(int argc, char const *argv[])
  5. {
  6.     key_t key;
  7.     int semid;
  8.     key = ftok("./app", 'b');
  9.     if (key < 0)
  10.     {
  11.         perror("ftok err");
  12.         return -1;
  13.     }
  14.     printf("%#x\n", key);
  15.     //创建或打开信号灯集
  16.     semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666);
  17.     if (semid < 0)
  18.     {
  19.         if (errno == EEXIST)
  20.             semid = semget(key, 2, 0666);
  21.         else
  22.         {
  23.             perror("semget err");
  24.             return -1;
  25.         }
  26.     }
  27.     else
  28.     {
  29.         //初始化
  30.         union semun sem;
  31.         sem.val = 10;
  32.         semctl(semid, 0, SETVAL, sem); //对编号为0的信号灯初值设置为10
  33.         sem.val = 0;
  34.         semctl(semid, 1, SETVAL, sem); //对编号为1的信号灯初值设置为0
  35.     }
  36.     printf("semid:%d\n", semid);
  37.     printf("%d\n",semctl(semid, 0, GETVAL));//获取编号为0的信号灯的值
  38.     printf("%d\n",semctl(semid, 1, GETVAL));//获取编号为1的信号灯的值
  39.     //pv操作
  40.     //p操作:申请资源
  41.     struct sembuf buf;
  42.     buf.sem_num = 0; //信号灯的编号
  43.     buf.sem_op = -1; //p操作
  44.     buf.sem_flg = 0; //阻塞
  45.     semop(semid, &buf, 1);
  46.     //v操作:释放资源
  47.     buf.sem_num = 1;
  48.     buf.sem_op = 1; //v操作
  49.     buf.sem_flg = 0;
  50.     semop(semid, &buf, 1);
  51.     printf("%d\n",semctl(semid, 0, GETVAL));//获取编号为0的信号灯的值
  52.     printf("%d\n",semctl(semid, 1, GETVAL));//获取编号为1的信号灯的值
  53.     //删除信号灯集
  54.     semctl(semid, 0, IPC_RMID); //指定任意一个编号即可删除信号灯集
  55.     return 0;
  56.    
复制代码
  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <sys/ipc.h>
  4. #include <sys/sem.h>
  5. #include <errno.h>
  6. union semun {
  7.     int val; //信号灯的初值
  8. };
  9. //初始化
  10. void seminit(int semid, int snum, int val)
  11. {
  12.     union semun sem;
  13.     sem.val = val;
  14.     semctl(semid, snum, SETVAL, sem);
  15. }
  16. //pv操作
  17. void sem_op(int semid, int num, int op)
  18. {
  19.     struct sembuf buf;
  20.     buf.sem_num = num;
  21.     buf.sem_op = op;
  22.     buf.sem_flg = 0;
  23.     semop(semid, &buf, 1);
  24. }
  25. int main(int argc, char const *argv[])
  26. {
  27.     key_t key;
  28.     int semid;
  29.     key = ftok("./app", 'b');
  30.     if (key < 0)
  31.     {
  32.         perror("ftok err");
  33.         return -1;
  34.     }
  35.     printf("%#x\n", key);
  36.     //创建或打开信号灯集
  37.     semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666);
  38.     if (semid < 0)
  39.     {
  40.         if (errno == EEXIST)
  41.             semid = semget(key, 2, 0666);
  42.         else
  43.         {
  44.             perror("semget err");
  45.             return -1;
  46.         }
  47.     }
  48.     else
  49.     {
  50.         //初始化
  51.         seminit(semid, 0, 10);
  52.         seminit(semid, 1, 0);
  53.     }
  54.     printf("semid:%d\n", semid);
  55.     printf("%d\n", semctl(semid, 0, GETVAL)); //获取编号为0的信号灯的值
  56.     printf("%d\n", semctl(semid, 1, GETVAL)); //获取编号为1的信号灯的值
  57.     //pv操作
  58.     //p操作:申请资源
  59.     sem_op(semid, 0, -1);
  60.     //v操作:释放资源
  61.     sem_op(semid, 1, 1);
  62.     printf("%d\n", semctl(semid, 0, GETVAL)); //获取编号为0的信号灯的值
  63.     printf("%d\n", semctl(semid, 1, GETVAL)); //获取编号为1的信号灯的值
  64.     //删除信号灯集
  65.     semctl(semid, 0, IPC_RMID);
  66.     return 0;
  67. }
复制代码
套接字

网络内容,笔者认为只写传输,脱离了我写这个目的(回顾,然后快速上手),所以放在了网络里面再写吧
其他补充

有名与无名管道的区别

无名管道有名管道特点只能在亲缘关系进程间使用半双工通信方式有固定的读端和写端,fd[0]:读,fd[1]:写端通过文件IO进行操作步骤:创建管道、读写操作不相关的任意进程间使用在路径中有管道文件,实际数据存在内核空间通过文件IO进行操作步骤:创建管道、打开管道、读写操作函数pipemkfifo读写特性当管道中没有数据,读阻塞当写满管道时,写阻塞当管道中没有数据,读阻塞当写满管道时,写阻塞IPC通讯的ftok函数

系统使用IPC通讯也就是消息队列,信号量,共享内存。这些操作都一个步骤就是使用这个函数接口ftok这个函数后面有2个参数const char *pathname, int proj_id第一个为我们创建的文件名,第二个为自己设置的ID值。
  1. key = ftok("./app.c", 'a');
  2.     if (key < 0)
  3.     {
  4.         perror("ftok err");
  5.         return -1;
  6.     }
  7.   printf("%#x\n", key);
  8. //打印结果为0x6130008e
复制代码
比如指定文件的索引节点号为65538,换算成16进制为0x010002,而你指定的ID值为38,换算成16进制为0x26,则最后的key_t返回值为0x26010002。
一个key值是由2部分确定,对应的key值才能通讯。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4