国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

面試必備 進程間的五種通信方式

網(wǎng)絡(luò) 通信技術(shù)
進程間通信(IPC)是指在不同進程間傳播或交換信息,它的方式通常有管道(包括無名管道和命名管道)、消息隊列、信號量、共享存儲、Socket、Streams等。

進程間通信(IPC,Inter-Process Communication)是指在不同進程間傳播或交換信息。

IPC的方式通常有管道(包括無名管道和命名管道)、消息隊列、信號量、共享存儲、Socket、Streams等。其中 Socket和Streams支持不同主機上的兩個進程IPC。

[[283876]]

一、管道

管道,通常指無名管道,是 UNIX 系統(tǒng)IPC最古老的形式。

1. 特點

  • 半雙工(數(shù)據(jù)流向僅有一個方向),具有固定的讀端和寫端
  • 只能用于父進程或兄弟線程之間通信(具有血緣關(guān)系的線程之間)
  • 一種特殊文件,可以用普通的read、write函數(shù)進行讀寫,但又不是普通文件,不屬于任何其它文件系統(tǒng),僅存在于內(nèi)存之中

2. 原型

  1. #include <unistd.h> 
  2. int pipe(int fd[2]); // 返回值:若成功返回0,失敗返回-1 

當一個管道建立時,它會創(chuàng)建兩個文件描述符:fd[0]為讀而打開,fd[1]為寫而打開。要關(guān)閉管道只需要關(guān)閉這兩個文件描述符即可。如下圖:

3. 例子

單個進程中的管道幾乎沒有任何用處。所以,通常調(diào)用 pipe 的進程接著調(diào)用 fork,這樣就創(chuàng)建了父進程與子進程之間的 IPC 通道。如下圖所示:

fork之后的半雙工管道

fork之后的半雙工管道

從父進程到子進程之間的管道

若要數(shù)據(jù)流從父進程流向子進程,則關(guān)閉父進程的讀端(fd[0])與子進程的寫端(fd[1]);反之,則可以使數(shù)據(jù)流從子進程流向父進程。

  1. #include<stdio.h> 
  2. #include<unistd.h> 
  3.  
  4. int main() 
  5.  int fd[2]; // 兩個文件描述符 
  6.  pid_t pid; 
  7.  char buff[20]; 
  8.  
  9.  if(pipe(fd) < 0) // 創(chuàng)建管道 
  10.  printf("Create Pipe Error!\n"); 
  11.  
  12.  if((pid = fork()) < 0) // 創(chuàng)建子進程 
  13.  printf("Fork Error!\n"); 
  14.  else if(pid > 0) // 父進程 
  15.  { 
  16.  close(fd[0]); // 關(guān)閉讀端 
  17.  write(fd[1], "hello world\n", 12); 
  18.  } 
  19.  else 
  20.  { 
  21.  close(fd[1]); // 關(guān)閉寫端 
  22.  read(fd[0], buff, 20); 
  23.  printf("%s", buff); 
  24.  } 
  25.  
  26.  return 0; 

二、命名管道(FIFO)

FIFO,也稱為命名管道,它是一種文件類型。

1. 特點

  • 與無名管道不同,命名管道可以在無關(guān)進程間通信
  • FIFO以一種特殊設(shè)備文件形式存在于文件系統(tǒng)中,有路徑名與之關(guān)聯(lián)

2. 原型

  1. #include <sys/stat.h> 
  2. int mkfifo(const char *pathname, mode_t mode); // 返回值:成功返回0,出錯返回-1 

其中的 mode 參數(shù)與下文中open函數(shù)中的 mode 相同

3. 例子

wirte:

  1. #include<stdio.h> 
  2. #include<stdlib.h> // exit 
  3. #include<fcntl.h> // O_WRONLY 
  4. #include<sys/stat.h> 
  5. #include<time.h> // time 
  6.  
  7. int main() 
  8.  int fd; 
  9.  int n, i; 
  10.  char buf[1024]; 
  11.  time_t tp; 
  12.  
  13.  printf("I am %d process.\n", getpid()); // 說明進程ID 
  14.  //當 open 一個FIFO時,是否設(shè)置非阻塞標志(O_NONBLOCK)的區(qū)別: 
  15.  
  16. //若沒有指定O_NONBLOCK(默認),只讀 open 要阻塞到某個其他進程為寫而打開此 FIFO。類似的,只寫 open 要阻塞到某個其他進程為讀而打開它。 
  17.  
  18. 若指定了O_NONBLOCK,則只讀 open 立即返回。而只寫 open 將出錯返回 -1 如果沒有進程已經(jīng)為讀而打開該 FIFO,其errno置ENXIO。 
  19.  if((fd = open("fifo1", O_WRONLY)) < 0) // 以寫打開一個FIFO  
  20.  { 
  21.  perror("Open FIFO Failed"); 
  22.  exit(1); 
  23.  } 
  24.  
  25.  for(i=0; i<10; ++i) 
  26.  { 
  27.  time(&tp); // 取系統(tǒng)當前時間 
  28.  n=sprintf(buf,"Process %d's time is %s",getpid(),ctime(&tp)); 
  29.  printf("Send message: %s", buf); // 打印 
  30.  if(write(fd, buf, n+1) < 0) // 寫入到FIFO中 
  31.  { 
  32.  perror("Write FIFO Failed"); 
  33.  close(fd); 
  34.  exit(1); 
  35.  } 
  36.  sleep(1); // 休眠1秒 
  37.  } 
  38.  
  39.  close(fd); // 關(guān)閉FIFO文件 
  40.  return 0; 

read:

  1. #include<stdio.h> 
  2. #include<stdlib.h> 
  3. #include<errno.h> 
  4. #include<fcntl.h> 
  5. #include<sys/stat.h> 
  6.  
  7. int main() 
  8.  int fd; 
  9.  int len; 
  10.  char buf[1024]; 
  11.  
  12.  if(mkfifo("fifo1", 0666) < 0 && errno!=EEXIST) // 創(chuàng)建FIFO管道 
  13.  perror("Create FIFO Failed"); 
  14.  
  15.  if((fd = open("fifo1", O_RDONLY)) < 0) // 以讀打開FIFO 
  16.  { 
  17.  perror("Open FIFO Failed"); 
  18.  exit(1); 
  19.  } 
  20.   
  21.  while((len = read(fd, buf, 1024)) > 0) // 讀取FIFO管道 
  22.  printf("Read message: %s", buf); 
  23.  
  24.  close(fd); // 關(guān)閉FIFO文件 
  25.  return 0; 

三、消息隊列

消息隊列,是消息的鏈接表,存放在內(nèi)核中。一個消息隊列由一個標識符(即隊列ID)來標識。

1. 特點

  • 消息隊列是面向記錄的,其中的消息具有特定的格式以及特定的優(yōu)先級
  • 消息隊列獨立于發(fā)送與接收進程。進程終止時,消息隊列及其內(nèi)容并不會被刪除
  • 消息隊列可以實現(xiàn)消息的隨機查詢, 消息不一定要以先進先出的次序讀取,也可以按消息的類型讀取

2. 原型

  1. #include <sys/msg.h> 
  2. // 創(chuàng)建或打開消息隊列:成功返回隊列ID,失敗返回-1 
  3. int msgget(key_t key, int flag); 
  4. // 添加消息:成功返回0,失敗返回-1 
  5. int msgsnd(int msqid, const void *ptr, size_t size, int flag); 
  6. // 讀取消息:成功返回消息數(shù)據(jù)的長度,失敗返回-1 
  7. int msgrcv(int msqid, void *ptr, size_t size, long type,int flag); 
  8. // 控制消息隊列:成功返回0,失敗返回-1 
  9. int msgctl(int msqid, int cmd, struct msqid_ds *buf); 

在以下兩種情況下,msgget將創(chuàng)建一個新的消息隊列:

  • 如果沒有與鍵值key相對應(yīng)的消息隊列,并且flag中包含了IPC_CREAT標志位。
  • key參數(shù)為IPC_PRIVATE。

函數(shù)msgrcv在讀取消息隊列時,type參數(shù)有下面幾種情況:

  • type == 0,返回隊列中的第一個消息;
  • type > 0,返回隊列中消息類型為 type 的第一個消息;
  • type < 0,返回隊列中消息類型值小于或等于 type 絕對值的消息,如果有多個,則取類型值最小的消息。

可以看出,type值非 0 時用于以非先進先出次序讀消息。也可以把 type 看做優(yōu)先級的權(quán)值。

3. 例子

msg_server:

  1. #include <stdio.h> 
  2. #include <stdlib.h> 
  3. #include <sys/msg.h> 
  4.  
  5. // 用于創(chuàng)建一個唯一的key 
  6. #define MSG_FILE "/etc/passwd" 
  7.  
  8. // 消息結(jié)構(gòu) 
  9. struct msg_form { 
  10.  long mtype; 
  11.  char mtext[256]; 
  12. }; 
  13.  
  14. int main() 
  15.  int msqid; 
  16.  key_t key; 
  17.  struct msg_form msg; 
  18.   
  19.  // 獲取key值 
  20.  if((key = ftok(MSG_FILE,'z')) < 0
  21.  { 
  22.  perror("ftok error"); 
  23.  exit(1); 
  24.  } 
  25.  
  26.  // 打印key值 
  27.  printf("Message Queue - Server key is: %d.\n", key); 
  28.  
  29.  // 創(chuàng)建消息隊列 
  30.  if ((msqid = msgget(key, IPC_CREAT|0777)) == -1) 
  31.  { 
  32.  perror("msgget error"); 
  33.  exit(1); 
  34.  } 
  35.  
  36.  // 打印消息隊列ID及進程ID 
  37.  printf("My msqid is: %d.\n", msqid); 
  38.  printf("My pid is: %d.\n", getpid()); 
  39.  
  40.  // 循環(huán)讀取消息 
  41.  for(;;)  
  42.  { 
  43.  msgrcv(msqid, &msg, 256, 888, 0);// 返回類型為888的第一個消息 
  44.  printf("Server: receive msg.mtext is: %s.\n", msg.mtext); 
  45.  printf("Server: receive msg.mtype is: %d.\n", msg.mtype); 
  46.  
  47.  msg.mtype = 999; // 客戶端接收的消息類型 
  48.  sprintf(msg.mtext, "hello, I'm server %d", getpid()); 
  49.  msgsnd(msqid, &msg, sizeof(msg.mtext), 0); 
  50.  } 
  51.  return 0; 

msg_client:

  1. #include <stdio.h> 
  2. #include <stdlib.h> 
  3. #include <sys/msg.h> 
  4.  
  5. // 用于創(chuàng)建一個唯一的key 
  6. #define MSG_FILE "/etc/passwd" 
  7.  
  8. // 消息結(jié)構(gòu) 
  9. struct msg_form { 
  10.  long mtype; 
  11.  char mtext[256]; 
  12. }; 
  13.  
  14. int main() 
  15.  int msqid; 
  16.  key_t key; 
  17.  struct msg_form msg; 
  18.  
  19.  // 獲取key值 
  20.  if ((key = ftok(MSG_FILE, 'z')) < 0)  
  21.  { 
  22.  perror("ftok error"); 
  23.  exit(1); 
  24.  } 
  25.  
  26.  // 打印key值 
  27.  printf("Message Queue - Client key is: %d.\n", key); 
  28.  
  29.  // 打開消息隊列 
  30.  if ((msqid = msgget(key, IPC_CREAT|0777)) == -1)  
  31.  { 
  32.  perror("msgget error"); 
  33.  exit(1); 
  34.  } 
  35.  
  36.  // 打印消息隊列ID及進程ID 
  37.  printf("My msqid is: %d.\n", msqid); 
  38.  printf("My pid is: %d.\n", getpid()); 
  39.  
  40.  // 添加消息,類型為888 
  41.  msg.mtype = 888
  42.  sprintf(msg.mtext, "hello, I'm client %d", getpid()); 
  43.  msgsnd(msqid, &msg, sizeof(msg.mtext), 0); 
  44.  
  45.  // 讀取類型為777的消息 
  46.  msgrcv(msqid, &msg, 256, 999, 0); 
  47.  printf("Client: receive msg.mtext is: %s.\n", msg.mtext); 
  48.  printf("Client: receive msg.mtype is: %d.\n", msg.mtype); 
  49.  return 0; 

四、信號量

信號量(semaphore)與已經(jīng)介紹過的 IPC 結(jié)構(gòu)不同,它是一個計數(shù)器。信號量用于實現(xiàn)進程間的互斥與同步,而不是用于存儲進程間通信數(shù)據(jù)。

1. 特點

  • 信號量用于進程間同步,若要在進程間傳遞數(shù)據(jù)需要結(jié)合共享內(nèi)存
  • 信號量基于操作系統(tǒng)的 PV 操作,程序?qū)π盘柫康牟僮鞫际窃硬僮?/li>
  • 每次對信號量的 PV 操作不僅限于對信號量值加 1 或減 1,而且可以加減任意正整數(shù)
  • 支持信號量組

2. 原型

最簡單的信號量是只能取 0 和 1 的變量,這也是信號量最常見的一種形式,叫做二值信號量(Binary Semaphore)。而可以取多個正整數(shù)的信號量被稱為通用信號量。

Linux 下的信號量函數(shù)都是在通用的信號量數(shù)組上進行操作,而不是在一個單一的二值信號量上進行操作。

  1. #include <sys/sem.h> 
  2. // 創(chuàng)建或獲取一個信號量組:若成功返回信號量集ID,失敗返回-1 
  3. int semget(key_t key, int num_sems, int sem_flags); 
  4. // 對信號量組進行操作,改變信號量的值:成功返回0,失敗返回-1 
  5. int semop(int semid, struct sembuf semoparray[], size_t numops);  
  6. // 控制信號量的相關(guān)信息 
  7. int semctl(int semid, int sem_num, int cmd, ...); 

當semget創(chuàng)建新的信號量集合時,必須指定集合中信號量的個數(shù)(即num_sems),通常為1; 如果是引用一個現(xiàn)有的集合,則將num_sems指定為 0 。

在semop函數(shù)中,sembuf結(jié)構(gòu)的定義如下:

  1. struct sembuf  
  2.  short sem_num; // 信號量組中對應(yīng)的序號,0~sem_nums-1 
  3.  short sem_op; // 信號量值在一次操作中的改變量 
  4.  short sem_flg; // IPC_NOWAIT, SEM_UNDO 

五、共享內(nèi)存

1. 特點

  • 共享內(nèi)存是最快的一種 IPC,因為進程是直接對內(nèi)存進行存取
  • 因為多個進程可以同時操作,所以需要進行同步
  • 信號量+共享內(nèi)存通常結(jié)合在一起使用,信號量用來同步對共享內(nèi)存的訪問

2. 原型

  1. #include <sys/shm.h> 
  2. // 創(chuàng)建或獲取一個共享內(nèi)存:成功返回共享內(nèi)存ID,失敗返回-1 
  3. int shmget(key_t key, size_t size, int flag); 
  4. // 連接共享內(nèi)存到當前進程的地址空間:成功返回指向共享內(nèi)存的指針,失敗返回-1 
  5. void *shmat(int shm_id, const void *addr, int flag); 
  6. // 斷開與共享內(nèi)存的連接:成功返回0,失敗返回-1 
  7. int shmdt(void *addr);  
  8. // 控制共享內(nèi)存的相關(guān)信息:成功返回0,失敗返回-1 
  9. int shmctl(int shm_id, int cmd, struct shmid_ds *buf); 

當用shmget函數(shù)創(chuàng)建一段共享內(nèi)存時,必須指定其 size;而如果引用一個已存在的共享內(nèi)存,則將 size 指定為0 。

當一段共享內(nèi)存被創(chuàng)建以后,它并不能被任何進程訪問。必須使用shmat函數(shù)連接該共享內(nèi)存到當前進程的地址空間,連接成功后把共享內(nèi)存區(qū)對象映射到調(diào)用進程的地址空間,隨后可像本地空間一樣訪問。

shmdt函數(shù)是用來斷開shmat建立的連接的。注意,這并不是從系統(tǒng)中刪除該共享內(nèi)存,只是當前進程不能再訪問該共享內(nèi)存而已。

shmctl函數(shù)可以對共享內(nèi)存執(zhí)行多種操作,根據(jù)參數(shù) cmd 執(zhí)行相應(yīng)的操作。常用的是IPC_RMID(從系統(tǒng)中刪除該共享內(nèi)存)。

 

責任編輯:趙寧寧 來源: 今日頭條
相關(guān)推薦

2018-05-30 13:58:02

Linux進程通信

2022-07-04 08:29:13

electron通信

2018-01-02 15:34:47

2017-08-06 00:05:18

進程通信開發(fā)

2010-01-05 10:00:48

Linux進程間通信

2019-05-15 08:00:00

vue組件間通信前端

2011-06-22 17:09:50

QT 進程 通信

2020-11-04 07:17:42

Nodejs通信進程

2019-05-08 11:10:05

Linux進程語言

2018-01-12 14:35:00

Linux進程共享內(nèi)存

2013-03-28 13:14:45

AIDL進程間通信Android使用AI

2017-06-19 13:36:12

Linux進程消息隊列

2019-10-18 09:00:00

開發(fā)工具軟件開發(fā)碼農(nóng)

2019-11-08 14:47:49

TCPIP網(wǎng)絡(luò)

2011-11-25 10:25:27

SpringJava

2011-06-24 14:01:34

Qt QCOP 協(xié)議

2019-06-04 09:00:00

Linux進程進程間通信

2021-11-02 11:35:17

通信進程面試

2020-11-18 09:06:04

Python

2009-06-19 18:26:38

Spring事務(wù)配置
點贊
收藏

51CTO技術(shù)棧公眾號

99视频这里有精品| 久久99久久人婷婷精品综合| 国产精品高清乱码在线观看| 欧美日韩亚洲在线观看| 久久五月激情| 成人午夜激情视频| 久久精品亚洲94久久精品| 亚洲最大色综合成人av| 国产网友自拍视频导航网站在线观看| 51vv免费精品视频一区二区 | 女人喷潮完整视频| 自由的xxxx在线视频| 欧美成人xxx| 中文在线播放一区二区| 亚洲影视资源网| 久久久在线视频| 无需播放器的av| 搜成人激情视频| 99pao成人国产永久免费视频| 亚洲婷婷在线视频| 91亚色免费| bt在线麻豆视频| 国产精品久久国产愉拍| 精品国产乱码久久久久久闺蜜 | 亚洲成人一区二区三区| 无遮挡的视频在线观看 | 手机在线观看国产精品| 国产高清一区在线观看| 93在线视频精品免费观看| 午夜久久电影网| 国产免费一区二区三区在线观看| 福利片免费在线观看| 成人午夜毛片| 91免费在线视频观看| 国产精品日韩在线播放| www.视频在线.com| 亚洲在线日韩| 这里只有精品在线观看| 热这里只有精品| 成人免费在线电影| 日韩成人免费电影| 亚洲国产日韩欧美在线图片| 日韩成人三级视频| 国产精品黄色片| 欧美少妇一区二区| 国产日韩在线| 五月婷婷六月综合| 精品999久久久| 玩弄japan白嫩少妇hd| 羞羞视频在线观看不卡| 亚洲欧美一区二区三区极速播放| 国产精品亚洲一区| 中文在线а√在线8| 一色桃子久久精品亚洲| 99国产在线| 欧美大电影免费观看| 国产精品人人做人人爽人人添| 国产suv精品一区二区| 国产精品免费精品自在线观看| 91精品国产aⅴ一区二区| 无码人妻丰满熟妇区96| 亚洲精品va| www.亚洲成人| 国产三级视频在线播放线观看| 精品制服美女久久| 国产精品专区h在线观看| 欧美自拍电影| 亚洲精品一区二区三区四区高清| 在线天堂日本| 啊啊啊啊啊啊啊视频在线播放| 国产精品婷婷| 久久久久这里只有精品| 第一页在线观看| 99久久精品国产导航| 国产脚交av在线一区二区| 加勒比久久高清| 麻豆一区二区在线观看| 青春草免费在线视频| 亚洲欧美日韩一区二区三区在线观看| 亚洲精品成人a8198a| 99热99re6国产在线播放| 91精品国模一区二区三区| 午夜视频国产| 亚洲色图欧洲色图| 亚洲乱码一区二区三区| av成人资源| 欧美精品久久久久久久免费观看| 国内高清免费在线视频| 亚洲一二三区视频在线观看| 正在播放91九色| 日韩综合一区| 在线观看视频亚洲| 精品176二区| 日韩欧美国产中文字幕| 头脑特工队2免费完整版在线观看| 久久婷婷综合激情| 国产小视频免费| 久久爱www久久做| 欧美日韩亚洲免费| 久久一区二区视频| 人妻少妇被粗大爽9797pw| 丰满白嫩尤物一区二区| 欧美精品一区在线| 欧美日韩三级电影在线| 欧美亚洲激情视频| 中文字幕日本一区| 欧美激情亚洲精品| 99ri日韩精品视频| 亚洲免费观看高清| 久久这里只精品| 国产河南妇女毛片精品久久久| 国产色视频在线播放| 午夜亚洲福利老司机| 欧美日韩视频在线播放| 91精品在线观看入口| 在线免费观看你懂的| 日韩欧美在线视频免费观看| av在线com| 9色porny自拍视频一区二区| 99re99| 无码av中文一区二区三区桃花岛| 久久综合之合合综合久久| 久久久999成人| 伊人久久综合| aaaaaa亚洲| 天天综合天天综合色| 97在线超碰| 51精品国产黑色丝袜高跟鞋 | 成人观看免费完整观看| 亚洲午夜三级在线| 激情视频网站在线播放色| 2019中文字幕在线免费观看| 新狼窝色av性久久久久久| 国产3p露脸普通话对白| 亚洲电影一区二区| 免费在线国产| 欧美成人精品在线播放| 亚洲国产高清一区二区三区| 久久综合久久网| 欧美日韩一区在线| 欧美123区| 亚洲专区国产精品| 日本va欧美va瓶| 在线视频中文字幕| 中文字幕在线亚洲| 伊人成人在线| 日本xxxxxx| 亚洲男女自偷自拍图片另类| 精品国产一区二区三区久久久樱花| 成人羞羞国产免费网站| 3atv一区二区三区| 国产一区网站| 免费一级网站| 国产精品黄视频| 亚洲在线中文字幕| theporn国产在线精品| 可以免费在线看黄的网站| 久久精品亚洲精品| 91亚洲大成网污www| 欧美高清你懂的| www一区二区www免费| 久久99久久99精品中文字幕| 国产精品午夜av| 国产激情美女久久久久久吹潮| 久久久精品免费免费| 成人开心激情| 欧美三日本三级少妇三99| 欧美性淫爽ww久久久久无| 国产免费播放一区二区| sihu成人| 久久久久久久亚洲精品| 97se狠狠狠综合亚洲狠狠| 都市激情亚洲一区| 色中文字幕在线观看| 精品夜色国产国偷在线| 国产精品自拍一区| 国产高清不卡| 成人免费观看在线| 久久久91精品国产| 2021中文字幕一区亚洲| www.91精品| 成人网免费视频| 国产福利视频一区| 欧美特级www| 国产精品日韩久久久| www.欧美日本韩国| 亚洲国产精品一区二区第一页 | 国产喂奶挤奶一区二区三区| 99久久久成人国产精品| 亚洲 高清 成人 动漫| 色妞欧美日韩在线| 久久久精品tv| 国内精品国产成人国产三级粉色| www.超碰com| 97超级碰碰人国产在线观看| 国产精品家庭影院| 精品不卡一区| 午夜视频成人| 欧美三级午夜理伦三级老人| 国产偷亚洲偷欧美偷精品|