【Linux】进程间通信IPC

目录

进程间通信 IPC

1. 进程间通信方式

2. 无名管道

2.1 特点

2.2 函数接口

2.3 注意事项

3. 有名管道

3.1 特点

3.2 函数接口

3.3 注意事项

3.4 有名管道和无名管道的区别

4. 信号

4.1概念

4.2信号的响应方式

4.3 信号种类

4.4 函数接口

4.4.1 信号发送和挂起

4.4.2 定时器

4.4.3 信号处理函数signal()

5. 共享内存

5.1特点

5.2步骤

5.3函数接口

5.4命令

6. 信号灯集

6.1 特点

6.2步骤

6.3 命令

6.4 函数接口

创建和使用信号灯:

函数操作:

把信号灯集加到共享内存:

7.消息队列:messagequeue

7.1特点

7.2步骤

7.3命令

7.4函数接口


进程间通信 IPC

InterProcess Communication

1. 进程间通信方式

1)早期的进程间通信:

无名管道(pipe)、有名管道(fifo)、信号(signal)

2)systerm V IPC:

共享内存(share memory)、消息队列(message queue)、信号灯集(semaphore set)

3)BSD:

套接字(socket)

2. 无名管道

2.1 特点

(1) 只能用于具有亲缘关系的进程之间的通信

(2) 半双工的通信模式,具有固定的读端fd[0]和写端fd[1]。

(3) 管道可以看成是一种特殊的文件,对于它的读写可以使用文件IO如read、write函数。

(4) 管道是基于文件描述符的通信方式。当一个管道建立时,它会创建两个文件描述符 fd[0]和fd[1]。其中fd[0]固定用于读管道,而fd[1]固定用于写管道。

2.2 函数接口

int pipe(int fd[2])功能:创建无名管道
参数:文件描述符 fd[0]:读端  fd[1]:写端
返回值:成功 0失败 -1
#include <stdio.h>
#include <unistd.h>int main(int argc, char const *argv[])
{char buf[65536] = "";int fd[2] = {0}; //fd[0]代表读端,fd[1]代表写端if (pipe(fd) < 0){perror("pipe err");return -1;}printf("%d %d\n", fd[0], fd[1]);//读写// write(fd[1], "hello", 32);// read(fd[0], buf, 32);// printf("%s\n", buf);//结构类似队列,先进先出//1. 当管道中没有数据时,读阻塞。// read(fd[0], buf, 32);// printf("%s\n", buf);//但是关闭写端就不一样//当管道中有数据,关闭写端可以读出数据来。无数据,关闭写端,读操作会立即返回(返回0)。//write(fd[1],"hello",5);// close(fd[1]);// read(fd[0],buf,5);// printf("%s\n",buf);//2.当管道这个写满数据时,写阻塞,管道空间大小是64K// write(fd[1], buf, 65536);// printf("write full\n");// write(fd[1],"a",1);         //阻塞,因为管道中被写满了已经。// printf("write after\n");//3.写满一次后,当管道中至少有4K空间时(也就是读了4K),才可以继续写,否则阻塞。//先写满一次再读// read(fd[0], buf, 4096); //如果读4095后面写就阻塞了,因为不到4K空间。// write(fd[1], "k", 1);// printf("write after\n");//4.当读端关闭,向管道中写入数据无意义,管道破裂,进程会收到内核发送的SIGNAL信号close(fd[0]);write(fd[1], "hello", 5);printf("read close!\n");return 0;
}

2.3 注意事项

(1) 当管道中无数据的时候,读操作会阻塞。

管道当中有数据,关闭写段,可以将数据读出。

管道中无数据,关闭写段,读操作会立即返回。

(2) 管道中写满时(管道大小64K)写操作会阻塞,写满一次时一旦有4K空间可以继续写。

(3) 只有管道的读端存在时,向管道中写数据才有意义,否则会导致管道破裂,向管道中写入数据进程将收到内核传来的SIGPIPE信号(通常时Broken pipe错误)。

练习:父子进程实现通信,父进程循环从终端输入数据,子进程循环打印数据,当输入quit结束。

提示:不需要加同步机制, 因为pipe无数据时读会阻塞。

考虑:创建管道是在fork之前还是之后?

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/wait.h>int main(int argc, char const *argv[])
{pid_t pid;char buf[32] = "";int fd[2] = {0};if (pipe(fd) < 0){perror("pipe err");return -1;}printf("%d %d\n", fd[0], fd[1]);pid = fork();if (pid < 0){perror("fokr err");return -1;}else if (pid == 0){//循环打印while (1){//read读管道中内容read(fd[0], buf, 32);//判断quit就breakif (strcmp(buf, "quit") == 0)break;//printf打印到终端printf("%s\n", buf);}}else{//循环输入while (1){//先scanfscanf("%s", buf);//将写入的buf中内容用write写进管道write(fd[1], buf, 32);//write(fd[1], buf, strlen(buf)+1);//+1是为了把\0也写进管道//判断quit就breakif (strcmp(buf, "quit") == 0)break;}}//回收子进程wait(NULL);return 0;
}

3. 有名管道

3.1 特点

1) 有名管道可以使互不相关的两个进程互相通信。

2) 有名管道可以通过路径名来指出,并且在文件系统中可见,但内容存放在内存中。但是读写数据不会存在文件中,而是在管道中。

3) 进程通过文件IO来操作有名管道

4) 有名管道遵循先进先出规则

5) 不支持如lseek() 操作

3.2 函数接口

int mkfifo(const char *filename,mode_t mode);
功能:创健有名管道
参数:filename:有名管道文件名mode:权限
返回值:成功:0失败:-1,并设置errno号注意对错误的处理方式:
如果错误是file exist时,注意加判断,如:if(errno == EEXIST)

注:函数只是在路径下创建管道文件,往管道中写的数据依然写在内核空间。

先创建有名管道,然后用文件IO操作:打开、读写和关闭。

3.3 注意事项

1) 只写方式打开阻塞,一直到另一个进程把读打开

2) 只读方式打开阻塞,一直到另一个进程把写打开

3) 可读可写,如果管道中没有数据,读阻塞

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>int main(int argc, char const *argv[])
{int fd;char buf[32] = "";if (mkfifo("./fifo", 0666) < 0){if (errno == EEXIST)printf("file exist\n");else{perror("mkfifo err");return -1;}}printf("mkfifo success\n");//打开管道文件fd = open("./fifo", O_RDWR);if (fd < 0){perror("open err");return -1;}//读写文件write(fd, "hello", 5);read(fd, buf, 5);printf("%s\n", buf);return 0;
}

练习:通过两个进程实现cp功能。

./input srcfile

./output destfile

input.c 读源文件

//创建有名管道

//打开管道文件,打开源文件

//循环读源文件,再把内容写道管道

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>int main(int argc, char const *argv[])
{int fd_fifo, fd_file;char buf[32] = "";ssize_t s;if (mkfifo("./fifo", 0666) < 0){if (errno == EEXIST)printf("file exist\n");else{perror("mkfifo err");return -1;}}printf("mkfifo success\n");//打开管道文件fd_fifo = open("./fifo", O_WRONLY);if (fd_fifo < 0){perror("open fifo err");return -1;}fd_file = open(argv[1], O_RDONLY);if (fd_file < 0){perror("open file err");return -1;}//读写while (1){//从文件读到buf,判断读不到就结束s=read(fd_file,buf,32);if(s==0)break;//把buf中数据写到有名管道中write(fd_fifo,buf,s);}close(fd_fifo);close(fd_file);return 0;
}

output.c 写目标文件

//创建有名管道

//打开有名管道,打开目标文件

//循环读管道,把内容写到目标文件

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>int main(int argc, char const *argv[])
{int fd_fifo, fd_file;char buf[32] = "";ssize_t s;if (mkfifo("./fifo", 0666) < 0){if (errno == EEXIST)printf("file exist\n");else{perror("mkfifo err");return -1;}}printf("mkfifo success\n");//打开管道文件fd_fifo = open("./fifo", O_RDONLY);if (fd_fifo < 0){perror("open fifo err");return -1;}fd_file = open(argv[1], O_WRONLY | O_CREAT | O_TRUNC, 0666);if (fd_file < 0){perror("open file err");return -1;}//读写while (1){//从有名管道中读数据到buf,判断s = read(fd_fifo, buf, 32);if (s == 0)break;//把buf中数据写到目标文件write(fd_file, buf, s);}close(fd_fifo);close(fd_file);return 0;
}

3.4 有名管道和无名管道的区别

无名管道

有名管道

使用场景

有亲缘关系的进程使用

不相干的两个进程使用

特点

半双工通讯方式

固定读端fd[0]和写端fd[1]

看做一种特殊文件,可以通过文件IO操作

在文件系统中会存在管道文件,数据存放在内核空间中

通过文件IO进行操作

不支持lseek操作,也遵循先进先出

函数

pipe()

直接read、write

mkfifo()

先打开open,再对管道文件read、write读写

读写特性

当管道中无数据,读阻塞

当管道中写满,写阻塞,直到有4K空间才可以写

读端关闭,写会导致管道破裂

只读方式打开会阻塞,直到另一个进程把写打开

只写方式打开会阻塞,直到另一个进程把读打开

可读可写,如果管道中没有数据,读阻塞。

4. 信号

kill -l: 显示系统中的信号

kill -num PID:给某个进程发送信号

4.1概念

1)信号是在软件层次上对中断机制的一种模拟,是一种异步通信方式

2)信号可以直接进行用户空间进程和内核进程之间的交互,内核进程也可以利用它来通知用户空间进程发生了哪些系统事件。

3)如果该进程当前并未处于执行态,则该信号就由内核保存起来,直到该进程恢复执行再传递给它;如果一个信号被进程设置为阻塞,则该信号的传递被延迟,直到其阻塞被取消时才被传递给进程。

信号的生命周期:

4.2信号的响应方式

1)忽略信号:对信号不做任何处理,但是有两个信号不能忽略:即SIGKILL及SIGSTOP。

2)捕捉信号:定义信号处理函数,当信号发生时,执行相应的处理函数。

3)执行缺省操作:Linux对每种信号都规定了默认操作

4.3 信号种类

SIGINT(2):中断信号,Ctrl-C 产生,用于中断进程

SIGQUIT(3):退出信号, Ctrl-\ 产生,用于退出进程并生成核心转储文件

SIGKILL(9):终止信号,用于强制终止进程。此信号不能被捕获或忽略。

SIGALRM(14):闹钟信号,当由 alarm() 函数设置的定时器超时时产生。

SIGTERM(15):终止信号,用于请求终止进程。此信号可以被捕获或忽略。termination

SIGCHLD(17):子进程状态改变信号,当子进程停止或终止时产生。

SIGCONT(18):继续执行信号,用于恢复先前停止的进程。

SIGSTOP(19):停止执行信号,用于强制停止进程。此信号不能被捕获或忽略。

SIGTSTP(20):键盘停止信号,通常由用户按下 Ctrl-Z 产生,用于请求停止进程。

4.4 函数接口

4.4.1 信号发送和挂起

#include <signal.h>
int kill(pid_t pid, int sig);
功能:信号发送
参数:pid:指定进程sig:要发送的信号
返回值:成功 0     失败 -1int raise(int sig);
功能:进程向自己发送信号
参数:sig:信号
返回值:成功 0   失败 -1相当于:kill(getpid(),sig);int pause(void);
功能:用于将调用进程挂起,直到收到信号为止。
#include <sys/types.h>
#include <signal.h>
#include <stdio.h>
#include <unistd.h>int main(int argc, char const *argv[])
{// //kill(getpid(), SIGKILL); //给指定进程发送信号,此例子指定当前进程// raise(SIGKILL);            //给自己发信号// while (1)//     ;pause(); //将进程挂起,直到收到信号为止,作用类似死循环但是不占用CPU。printf("hello\n");return 0;
}

4.4.2 定时器

unsigned int alarm(unsigned int seconds)功能:在进程中设置一个定时器。当定时器指定的时间到了时,它就向进程发送SIGALARM信号。
参数:seconds:定时时间,单位为秒
返回值:如果调用此alarm()前,进程中已经设置了闹钟时间,则返回上一个闹钟时间的剩余时间,否则返回0。
注意:一个进程只能有一个闹钟时间。如果在调用alarm时已设置过闹钟时间,则之前的闹钟时间被新值所代替。
常用操作:取消定时器alarm(0),返回旧闹钟余下秒数。
#include <sys/types.h>
#include <signal.h>
#include <stdio.h>
#include <unistd.h>int main(int argc, char const *argv[])
{printf("%d\n", alarm(10));sleep(1);printf("%d\n", alarm(3));pause(); //将进程挂起,直到收到信号为止,作用类似死循环但是不占用CPU。return 0;
}

系统默认对SIGALRM(闹钟到点后内核发送的信号)信号的响应: 如果不对SIGALRM信号进行捕捉或采取措施,默认情况下,闹钟响铃时刻会退出进程。例如:

#include <sys/types.h>
#include <signal.h>
#include <stdio.h>
#include <unistd.h>int main(int argc, char const *argv[])
{printf("%d\n", alarm(3));while(1);  //闹钟响铃后结束进程return 0;
}

4.4.3 信号处理函数signal()

typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
功能:信号处理函数
参数:signum:要处理的信号handler:信号处理方式SIG_IGN:忽略信号  (忽略 ignore)SIG_DFL:执行默认操作 (默认 default)handler:捕捉信号 (handler为函数名,可以自定义)void handler(int sig){} //函数名可以自定义, 参数为要处理的信号返回值:成功:设置之前的信号处理方式失败:-1

补充:typedef给数据类型重命名

#include <stdio.h>//给普通数据类型int重命名
typedef int size4;        //给指针类型int* 重命名
typedef int *int_p;       //给数组类型int [10]重命名
typedef int intArr10[10]; //给函数指针void (*)()重命名
typedef void (*fun_p)(); void fun()
{printf("fun\n");
}int main(int argc, char const *argv[])
{size4 a = 10;             //相当于int aint_p p = &a;             //相当于int* pintArr10 arr = {1, 2, 3}; //相当于int arr[10]fun_p fp = fun;           //相当于void (*pf)();printf("%d\n", *p);printf("%d\n", arr[0]);fp();return 0;
}

例子:

#include <signal.h>
#include <stdio.h>
#include <unistd.h>void handler(int sig)
{printf("ctrl+C\n");
}int main(int argc, char const *argv[])
{//signal(SIGINT,SIG_IGN);   //忽略信号//signal(SIGINT,SIG_DFL);   //按默认方式处理信号signal(SIGINT,handler);     //捕捉信号,比较常用的方式while(1); //为了不让进程结束return 0;
}

练习:

用信号的知识实现司机和售票员问题。

1)售票员捕捉SIGINT(代表开车)信号,向司机发送SIGUSR1信号,司机打印(let's gogogo)

2)售票员捕捉SIGQUIT(代表停车)信号,向司机发送SIGUSR2信号,司机打印(stop the bus)

3)司机捕捉SIGTSTP(代表到达终点站)信号,向售票员发送SIGUSR1信号,售票员打印(please get off the bus)

4)司机等待售票员下车,之后司机再下车。

分析:司机(父进程)、售票员(子进程)

售票员:捕捉:SIGINT SIGQUIT SIGUSR1

忽略:SIGTSTP

司机:捕捉:SIGUSR1 SIGUSR2 SIGTSTP

忽略:SIGINT SIGQUIT

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>pid_t pid;void saler(int sig)
{if (sig == SIGINT)kill(getppid(),SIGUSR1);else if(sig == SIGQUIT)kill(getppid(),SIGUSR2);else if(sig == SIGUSR1){printf("pls get off the bus!\n");exit(0);}
}void driver(int sig)
{if(sig == SIGUSR1)printf("[driver] let's gogogo!\n");else if(sig == SIGUSR2)printf("stop the bus!\n");else if(sig == SIGTSTP){kill(pid,SIGUSR1);wait(NULL);         //等售票员下车以后再下车exit(0);        }
}int main(int argc, char const *argv[])
{if ((pid = fork()) < 0){perror("fokr err");return -1;}else if (pid == 0){printf("hi,i am saler %d\n", getpid());signal(SIGINT,saler);signal(SIGQUIT,saler);signal(SIGUSR1,saler);signal(SIGTSTP,SIG_IGN);}else{printf("hi, i am driver %d\n", getpid());signal(SIGUSR1,driver);signal(SIGUSR2,driver);signal(SIGTSTP,driver);signal(SIGINT,SIG_IGN);signal(SIGQUIT,SIG_IGN);}while(1)pause(); //不能只发送一个信号进程就结束了,所以可以循环挂起,不占用CPU。return 0;
}

5. 共享内存

5.1特点

1)共享内存是一种最为高效的进程间通信方式,进程可以直接读写内存,而不需要任何数据的拷贝。

2)为了在多个进程间交换信息,内核专门留出了一块内存区,可以由需要访问的进程将其映射到自己的私有地址空间。

3)进程就可以直接读写这一内存区而不需要进行数据的拷贝,从而大大提高的效率。

4)由于多个进程共享一段内存,因此也需要依靠某种同步机制,如互斥锁和信号量等

5.2步骤

a.创建key值

b.创建或者打开共享内存

c.映射共享内存到用户空间

d.撤销映射

e.删除共享内存

5.3函数接口

key_t ftok(const char *pathname, int proj_id);
功能:创建出来的具有唯一映射关系的一个key值,帮助操作系统用来标识一块共享内存
参数:Pathname:已经存在的可访问文件的名字Proj_id:一个字符(因为只用低8位)
返回值:成功:key值失败:-1int shmget(key_t key, size_t size, int shmflg);
功能:创建或打开共享内存
参数:key  键值size   共享内存的大小shmflg   IPC_CREAT|IPC_EXCL|0777注意:shmflg为IPC_CREAT|IPC_EXCL|0777这种形式代表创建共享内存,如果只有权限代表打开共享内存返回值:成功  shmid出错    -1void  *shmat(int  shmid,const  void  *shmaddr,int  shmflg); //attaches
功能:映射共享内存,即把指定的共享内存映射到进程的地址空间用于访问
参数:shmid   共享内存的id号shmaddr   一般为NULL,表示由系统自动完成映射如果不为NULL,那么有用户指定shmflg:SHM_RDONLY就是对该共享内存只进行读操作0     可读可写
返回值:成功:完成映射后的地址,出错:-1(地址)
用法:if((p = (char *)shmat(shmid,NULL,0)) == (char *)-1)int shmdt(const void *shmaddr); //detaches
功能:取消映射
参数:要取消的地址
返回值:成功0  失败的-1int  shmctl(int  shmid,int  cmd,struct shmid_ds *buf); //control
功能:(删除共享内存),对共享内存进行各种操作
参数:shmid   共享内存的id号cmd     IPC_STAT 获得shmid属性信息,存放在第三参数IPC_SET 设置shmid属性信息,要设置的属性放在第三参数IPC_RMID:删除共享内存,此时第三个参数为NULL即可buf    shmid所指向的共享内存的地址,空间被释放以后地址就赋值为null
返回:成功0 失败-1用法:shmctl(shmid,IPC_RMID,NULL);
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h>int main(int argc, char const *argv[])
{key_t key;int shmid;char *p;//创建key值,key值由指定文件的inode号和字符的ascii码组合而成key = ftok("./shm.c", '6');if (key < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//打开或创建共享内存shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666); //创建共享内存if (shmid <= 0){if (errno == EEXIST)shmid = shmget(key, 128, 0666); //直接打开已经存在的共享内存else{perror("shmget err");return -1;}}printf("shmid: %d\n", shmid);//映射共享内存p = (char *)shmat(shmid, NULL, 0);if (p == (char *)-1){perror("shmat err");return -1;}//操作共享内存strcpy(p, "hello");printf("%s\n", p);//取消映射shmdt(p);// //删除共享内存// shmctl(shmid,IPC_RMID,NULL);return 0;
}

5.4命令

ipcs-m:查看系统中的共享内存

ipcrm-mshmid:删除指定共享内存

注意:可能不能删除还存在进程使用的共享内存,这时候可以用kill杀死多余进程,再使用ipcs查看。

练习:两个进程实现通信,一个进程循环从终端输入,另一个进程循环打印,当输入quit时结束。

提示:为了共享标志位可以和buf封装到一个结构体里作为共享内存。

struct msg

{

int flag;

char buf[32];

};

输入进程:

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h>typedef struct msg
{int flag;char buf[32];
} msg_t;int main(int argc, char const *argv[])
{key_t key;int shmid;msg_t *p;//创建key值,key值由指定文件的inode号和字符的ascii码组合而成key = ftok("./shm.c", '6');if (key < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//打开或创建共享内存shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666); //创建共享内存if (shmid <= 0){if (errno == EEXIST)shmid = shmget(key, 128, 0666); //直接打开已经存在的共享内存else{perror("shmget err");return -1;}}printf("shmid: %d\n", shmid);//映射共享内存p = (msg_t *)shmat(shmid, NULL, 0);if (p == (msg_t *)-1){perror("shmat err");return -1;}p->flag = 0; //初始化//操作共享内存while (1){scanf("%s", p->buf);p->flag = 1;if (strcmp(p->buf, "quit") == 0)break;}//取消映射shmdt(p);// //删除共享内存// shmctl(shmid,IPC_RMID,NULL);return 0;
}

输出进程:

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h>typedef struct msg
{int flag;char buf[32];
} msg_t;int main(int argc, char const *argv[])
{key_t key;int shmid;msg_t *p;//创建key值,key值由指定文件的inode号和字符的ascii码组合而成key = ftok("./shm.c", '6');if (key < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//打开或创建共享内存shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666); //创建共享内存if (shmid <= 0){if (errno == EEXIST)shmid = shmget(key, 128, 0666); //直接打开已经存在的共享内存else{perror("shmget err");return -1;}}printf("shmid: %d\n", shmid);//映射共享内存p = (msg_t *)shmat(shmid, NULL, 0);if (p == (msg_t *)-1){perror("shmat err");return -1;}//操作共享内存p->flag = 0;while (1){if (strcmp(p->buf, "quit") == 0)break;if (p->flag == 1){printf("%s\n", p->buf);p->flag = 0;}}//取消映射shmdt(p);// //删除共享内存// shmctl(shmid,IPC_RMID,NULL);return 0;
}

6. 信号灯

线程:全局变量,或者通过信号量来实现同步

初始化:sem_init(&sem,0,0);

申请资源:sem_wait(&sem);p操作-1

释放资源:sem_post(&sem);v操作+1

6.1 特点

信号灯(semaphore),也叫信号量,信号灯集是一个信号灯的集合。它是不同进程间或一个给定进程内部不同线程间同步的机制;

而Posix信号灯指的是单个计数信号灯:无名信号灯、有名信号灯。(咱们学的是无名信号灯)

System V的信号灯是一个或者多个信号灯的一个集合。其中的每一个都是单独的计数信号灯。

通过信号灯集实现共享内存的同步操作。

6.2步骤

(1) 创建或者打开信号灯集:semget

(2) 初始化信号灯:semctl

(3) PV操作:semop

(4) 删除信号灯集:semctl

6.3 命令

ipcs -s查看系统中信号灯集

ipcrm -s semid:删除信号灯集

注意:有时候可能会创建失败,或者semid为0,所以创建完可以用命令看看,如果为0删除了重新创建就可以了。

6.4 函数接口

int semget(key_t key, int nsems, int semflg);
功能:创建/打开信号灯
参数:key:ftok产生的key值nsems:信号灯集中包含的信号灯数目semflg:信号灯集的访问权限,通常为IPC_CREAT|IPC_EXCL|0666返回值:成功:信号灯集ID失败:-1int semctl ( int semid, int semnum,  int cmd…/*union semun arg*/);
功能:信号灯集合的控制(初始化/删除)
参数:semid:信号灯集IDsemnum: 要操作的集合中的信号灯编号,信号灯编号从0开始cmd: GETVAL:获取信号灯的值,返回值是获得值SETVAL:设置信号灯的值,需要用到第四个参数:共用体IPC_RMID:从系统中删除信号灯集合
返回值:成功 0失败 -1用法:
1. 初始化信号灯集:
需要自定义共用体
union semun{int val;
} mysemun;
mysemun.val = 10;
semctl(semid, 0, SETVAL, mysemun);
2. 获取信号灯值:函数semctl(semid, 0, GETVAL)的返回值
3. 删除信号灯集:semctl(semid, 0, IPC_RMID);int semop ( int semid, struct sembuf  *opsptr,  size_t  nops);
功能:对信号灯集合中的信号量进行PV操作
参数:semid:信号灯集IDopsptr:操作方式nops:  要操作的信号灯的个数 1个
返回值:成功 :0失败:-1struct sembuf {short  sem_num; // 要操作的信号灯的编号short  sem_op;  //    0 :  等待,直到信号灯的值变成0//   1  :  释放资源,V操作//   -1 :  申请资源,P操作                    short  sem_flg; // 0(阻塞),IPC_NOWAIT, SEM_UNDO};
注意:直接用这个结构体定义变量就可以了,结构体不需要自己写。用法:
申请资源 P操作:mysembuf.sem_num = 0;
mysembuf.sem_op = -1;
mysembuf.sem_flg = 0;
semop(semid, &mysembuf, 1);
释放资源 V操作:mysembuf.sem_num = 0;
mysembuf.sem_op = 1;
mysembuf.sem_flg = 0;
semop(semid, &mysembuf, 1);

创建使用信号灯

#include <sys/types.h>
#include <sys/ipc.h>
#include <stdio.h>
#include <sys/sem.h>
#include <errno.h>union semun
{int val;
};int main(int argc, char const *argv[])
{int semid;key_t key;if ((key = ftok("./sem.c", 66)) < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//创建信号灯集semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666);if (semid <= 0){if (errno == EEXIST)semid = semget(key, 2, 0666);   //直接打开信号灯集else{perror("semget err");return -1;}}else  //确保对信号灯集中的信号灯只初始化一次,下次执行程序还继续获得上一次的值。如果想修改初值,需要删除信号灯然后重新打开。{union semun sem;sem.val = 10;semctl(semid,0,SETVAL,sem); //对编号为0的信号灯初值设为10sem.val = 0;semctl(semid,1,SETVAL,sem); //对编号为1的信号灯初值设为0}printf("semid: %d\n", semid);//获取信号灯初值printf("%d\n",semctl(semid,0,GETVAL)); //获取编号为0的信号灯的值printf("%d\n",semctl(semid,1,GETVAL)); //获取编号为1的信号灯的值//pv操作struct sembuf buf;buf.sem_num=0;buf.sem_op=-1; //申请资源,P操作,-1buf.sem_flg=0; //阻塞semop(semid,&buf,1);buf.sem_num=1;buf.sem_op=1; //释放资源,V操作,+1buf.sem_flg=0;semop(semid,&buf,1);printf("%d\n",semctl(semid,0,GETVAL)); //获取编号为0的信号灯的值printf("%d\n",semctl(semid,1,GETVAL)); //获取编号为1的信号灯的值// //删除信号灯集// semctl(semid,0,IPC_RMID); //指定任意一个信号灯就可以,会删除整个信号灯集。return 0;
}

函数操作:

#include <sys/types.h>
#include <sys/ipc.h>
#include <stdio.h>
#include <sys/sem.h>
#include <errno.h>union semun {int val;
};void seminit(int semid, int num, int val) //初始化
{union semun sem;sem.val = val;semctl(semid, num, SETVAL, sem); //对编号为num的信号灯初值设为val
}void sem_op(int semid, int num,int op)
{struct sembuf buf;buf.sem_num = num;buf.sem_op = op;buf.sem_flg =0;semop(semid,&buf,1);
}int main(int argc, char const *argv[])
{int semid;key_t key;if ((key = ftok("./sem.c", 66)) < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//创建信号灯集semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666);if (semid <= 0){if (errno == EEXIST)semid = semget(key, 2, 0666); //直接打开信号灯集else{perror("semget err");return -1;}}else //确保对信号灯集中的信号灯只初始化一次,下次执行程序还继续获得上一次的值。如果想修改初值,需要删除信号灯然后重新打开。{seminit(semid, 0, 10);seminit(semid, 1, 0);}printf("semid: %d\n", semid);//获取信号灯初值printf("%d\n", semctl(semid, 0, GETVAL)); //获取编号为0的信号灯的值printf("%d\n", semctl(semid, 1, GETVAL)); //获取编号为1的信号灯的值//pv操作sem_op(semid,0,-1);sem_op(semid,1,1);printf("%d\n", semctl(semid, 0, GETVAL)); //获取编号为0的信号灯的值printf("%d\n", semctl(semid, 1, GETVAL)); //获取编号为1的信号灯的值// //删除信号灯集// semctl(semid,0,IPC_RMID); //指定任意一个信号灯就可以,会删除整个信号灯集。return 0;
}

信号灯加到共享内存

输入功能程序:

#include <sys/types.h>
#include <sys/ipc.h>
#include <stdio.h>
#include <sys/sem.h>
#include <errno.h>
#include <string.h>
#include <sys/shm.h>union semun {int val;
};void seminit(int semid, int num, int val) //初始化
{union semun sem;sem.val = val;semctl(semid, num, SETVAL, sem); //对编号为num的信号灯初值设为val
}void sem_op(int semid, int num, int op)
{struct sembuf buf;buf.sem_num = num;buf.sem_op = op;buf.sem_flg = 0;semop(semid, &buf, 1);
}int main(int argc, char const *argv[])
{int semid;int shmid;char *p;key_t key;if ((key = ftok("./sem.c", 66)) < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//创建/打开共享内存shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666);if (shmid <= 0){if (errno == EEXIST)shmid = shmget(key, 128, 0666);else{perror("shmget err");return -1;}}printf("shmid: %d\n", shmid);//创建信号灯集semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666);if (semid <= 0){if (errno == EEXIST)semid = semget(key, 2, 0666); //直接打开信号灯集else{perror("semget err");return -1;}}else //确保对信号灯集中的信号灯只初始化一次,下次执行程序还继续获得上一次的值。如果想修改初值,需要删除信号灯然后重新打开。{seminit(semid, 0, 0);}printf("semid: %d\n", semid);//共享内存映射p = (char *)shmat(shmid, NULL, 0);if (p == (char *)-1){perror("shmat err");return -1;}while (1){scanf("%s", p);//释放资源sem_op(semid,0,1);if (strcmp(p, "quit")==0)break;}shmdt(p);shmctl(shmid,IPC_RMID,NULL);// //删除信号灯集// semctl(semid,0,IPC_RMID); //指定任意一个信号灯就可以,会删除整个信号灯集。return 0;
}

out.c

#include <sys/types.h>
#include <sys/ipc.h>
#include <stdio.h>
#include <sys/sem.h>
#include <errno.h>
#include <string.h>
#include <sys/shm.h>union semun {int val;
};void seminit(int semid, int num, int val) //初始化
{union semun sem;sem.val = val;semctl(semid, num, SETVAL, sem); //对编号为num的信号灯初值设为val
}void sem_op(int semid, int num, int op)
{struct sembuf buf;buf.sem_num = num;buf.sem_op = op;buf.sem_flg = 0;semop(semid, &buf, 1);
}int main(int argc, char const *argv[])
{int semid;int shmid;char *p;key_t key;if ((key = ftok("./sem.c", 66)) < 0){perror("ftok err");return -1;}printf("key: %#x\n", key);//创建/打开共享内存shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666);if (shmid <= 0){if (errno == EEXIST)shmid = shmget(key, 128, 0666);else{perror("shmget err");return -1;}}printf("shmid: %d\n", shmid);//创建信号灯集semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666);if (semid <= 0){if (errno == EEXIST)semid = semget(key, 2, 0666); //直接打开信号灯集else{perror("semget err");return -1;}}else //确保对信号灯集中的信号灯只初始化一次,下次执行程序还继续获得上一次的值。如果想修改初值,需要删除信号灯然后重新打开。{seminit(semid, 0, 0);}printf("semid: %d\n", semid);//共享内存映射p = (char *)shmat(shmid, NULL, 0);if (p == (char *)-1){perror("shmat err");return -1;}while (1){sem_op(semid,0,-1); //申请资源if (strcmp(p, "quit")==0)break;printf("%s\n", p);}shmdt(p);shmctl(shmid, IPC_RMID, NULL);// //删除信号灯集// semctl(semid,0,IPC_RMID); //指定任意一个信号灯就可以,会删除整个信号灯集。return 0;
}

7.消息队列:messagequeue

传统:无名管道、有名管道、信号

systemV:共享内存、信号灯集、消息队列

按消息的类型添加或读取消息

队列原则

7.1特点

消息队列是IPC对象(活动在内核级别的一种进程间通信的工具)的一种

一个消息队列由一个标识符 (即队列ID)来标识

消息队列就是一个消息的列表。用户可以在消息队列中添加消息、读取消息等

消息队列可以按照类型(自己设一个值作为类型)来发送/接收消息

7.2步骤

(1) 产生key值ftok

(2) 创建或打开消息队列msgget()

(3) 添加消息:按照消息的类型添加到已经打开的消息队列的末尾msgsnd()

(4) 读取消息:可以按照消息的类型从消息队列中读走msgrcv()

(5) 删除消息队列:msgctl()

7.3命令

ipcs -q:查看系统中的消息队列

ipcrm -q msgid:删除消息队列

7.4函数接口

int msgget(key_t key, int flag);
功能:创建或打开一个消息队列
参数:  key值flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666返回值:成功:msgid失败:-1int msgsnd(int msqid, const void *msgp, size_t size, int flag); 
功能:添加消息
参数:msqid:消息队列的IDmsgp:指向消息的指针。常用消息结构msgbuf如下:struct msgbuf{long mtype;          //消息类型char mtext[N]};     //消息正文size:发送的消息正文的字节数flag:IPC_NOWAIT消息没有发送完成函数也会立即返回    0:直到发送完成函数才返回
返回值:成功:0失败:-1使用:msgsnd(msgid, &msg,sizeof(msg)-sizeof(long), 0)注意:消息结构除了第一个成员必须为long类型外,其他成员可以根据应用的需求自行定义。int msgrcv(int msgid,  void* msgp,  size_t  size,  long msgtype,  int  flag);
功能:读取消息
参数:msgid:消息队列的IDmsgp:存放读取消息的空间size:接受的消息正文的字节数(sizeof(msgp)-sizeof(long))msgtype:0:接收消息队列中第一个消息。大于0:接收消息队列中第一个类型为msgtyp的消息.小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。flag:0:若无消息函数会一直阻塞IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG
返回值:成功:接收到的消息的长度失败:-1int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
功能:对消息队列的操作,删除消息队列
参数:msqid:消息队列的队列IDcmd:IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。IPC_SET:设置消息队列的属性。这个值取自buf参数。IPC_RMID:从系统中删除消息队列。buf:消息队列缓冲区
返回值:成功:0失败:-1用法:msgctl(msgid, IPC_RMID, NULL)
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdio.h>
#include <errno.h>struct msgbuf
{long type; //必须有而且必须是long类型,表示消息类型,值>0int num;   //正文随便定义char ch;
};int main(int argc, char const *argv[])
{key_t key;int msgid;key = ftok("msg.c", 99);if (key < 0){perror("ftok err");return -1;}printf("key:%#x\n", key);//打开消息队列msgid = msgget(key, IPC_CREAT | IPC_EXCL | 0666);if (msgid <= 0){if (errno == EEXIST)msgid = msgget(key, 0666);else{perror("msgget err");return -1;}}printf("msgid:%d\n", msgid);//添加消息struct msgbuf msg;msg.type = 10;msg.num = 100;msg.ch = 'a';msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0); //0:阻塞等发完才返回msg.type = 20;msg.num = 200;msg.ch = 'b';msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0);//读取消息struct msgbuf m;msgrcv(msgid, &m, sizeof(m) - sizeof(long), 20, 0); //读取消息队列中类型为20的第一个消息printf("%d %c\n", m.num, m.ch);msgctl(msgid,IPC_RMID,NULL);return 0;
}

两个进程分别发和收:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/2639.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringMVC——原理简介

狂神SSM笔记 DispatcherServlet——SpringMVC 的核心 SpringMVC 围绕DispatcherServlet设计。 DispatcherServlet的作用是将请求分发到不同的处理器&#xff08;即不同的Servlet&#xff09;。根据请求的url&#xff0c;分配到对应的Servlet接口。 当发起请求时被前置的控制…

openssl s_server源码剥离

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github&#xff1a;codetoys&#xff0c;所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的&#xff0c;可以在任何平台上使用。 源码指引&#xff1a;github源…

算法库里的heap算法,仿函数和模版进阶(续)

文章目录 算法库里面的heap仿函数模版非类型模版参数array特化函数模版的特化类模版的特化 分离编译 算法库里面的heap sort_heap是算法库里的函数&#xff0c;前提要求是堆才能排序is_heap判断是否是堆make_heap建堆算法 int main() {int a[5] { 10,19,27,39,19 };std::vec…

具身导航如何利用取之不尽的网络视频资源!RoomTour3D:基于几何感知的视频-指令训练调优

作者&#xff1a;Mingfei Han, Liang Ma, Kamila Zhumakhanova, Ekaterina Radionova, Jingyi Zhang, Xiaojun Chang, Xiaodan Liang, Ivan Laptev 单位&#xff1a;穆罕默德本扎耶德人工智能大学计算机视觉系&#xff0c;中山大学深圳校区&#xff0c;悉尼科技大学ReLER实验室…

解决报错:未定义标识符 “M_PI“

问题&#xff1a; 使用C编译&#xff0c;已经用#include <cmath>包含了头文件&#xff0c;但是在使用M_PI时依旧报错说未定义 原因&#xff1a; 在某些编译器中&#xff0c;<cmath> 库中的 M_PI 是一个条件宏&#xff0c;需要 _USE_MATH_DEFINES 宏被定义才能使用。…

TensorFlow深度学习实战(5)——神经网络性能优化技术详解

TensorFlow深度学习实战&#xff08;5&#xff09;——神经网络性能优化技术详解 0. 前言1. 识别 MNIST 手写数字1.1 MNIST 数据集1.2 独热编码1.3 定义神经网络1.4 训练神经网络 2. 构建深度神经网络3. 添加 Dropout 提高模型泛化能力4. 不同优化器对模型性能的影响5. 训练 ep…

代码随想录算法训练营day31

代码随想录算法训练营 —day31 文章目录 代码随想录算法训练营前言一、 56. 合并区间二、738. 单调递增的数字三、968.监控二叉树总结 前言 今天是算法营的第31天&#xff0c;希望自己能够坚持下来&#xff01; 今日任务&#xff1a; ● 56. 合并区间 ● 738.单调递增的数字 …

通过maven命令上传jar包至nexus v3.7.1

1 nexus和maven的简介 1.1 nexus ‌Nexus‌是由Sonatype公司开发的一款强大的制品仓库管理软件&#xff0c;主要用于搭建和管理各种类型的仓库&#xff0c;包括Maven、NuGet、npm等。Nexus支持多种仓库类型&#xff0c;如代理仓库&#xff08;代理互联网中的中央仓库&#xf…

level(三) filterblock

filterblock用于确定某个key是否存在于某个datablock中&#xff0c;在插入一个key到datablock中时也会插入一个key到filterblock中&#xff0c;filterblock中会记录所有的key&#xff0c;并通过布隆过滤器来确定一个key是否存于这个datablock中。下面来看下filterblock的代码&a…

优化 Vue项目中 app.js 文件过大,初始化加载过慢、带宽占用过大等问题

已亲测&#xff0c;绝对有效&#xff0c;底部有改善前后对比图证明。 1.服务器 nginx 增加配置 #开启gzip压缩 gzip on; #设置gzip压缩级别&#xff0c;2级是性价比最高的 gzip_comp_level 2; #设置动态gzip压缩的文件类型 gzip_types text/plain text/css text/javascript a…

浅谈云计算16 | 存储虚拟化技术

存储虚拟化技术 一、块级存储虚拟化基础2.1 LUN 解析2.1.1 LUN 概念阐释2.1.2 LUN 功能特性 2.2 Thick LUN与Thin LUN2.2.1 Thick LUN特性剖析2.2.2 Thin LUN特性剖析 三、块级存储虚拟化技术实现3.1 基于主机的实现方式3.1.1 原理阐述3.1.2 优缺点评估 3.2 基于存储设备的实现…

手摸手实战前端项目CI CD

由于图片和格式解析问题&#xff0c;为了更好阅读体验可前往 阅读原文 CI/CD 是 持续集成&#xff08;Continuous Integration&#xff09; 和 持续交付/部署&#xff08;Continuous Delivery/Continuous Deployment&#xff09; 的缩写&#xff0c;是现代软件开发中的一种自动…

【EI 会议征稿通知】第七届机器人与智能制造技术国际会议 (ISRIMT 2025)

第七届机器人与智能制造技术国际会议 (ISRIMT 2025) 2025 7th International Symposium on Robotics & Intelligent Manufacturing Technology 会议主要围绕“机器人”、“智能制造技术” 等研究领域展开讨论&#xff0c;旨在为机器人与智能制造技术等领域的专家学者、工…

【Linux】信号

目录 一、信号的概念二、信号的产生2.1 通过键盘进行信号的产生2.2 通过系统调用进行信号的产生2.2.1 kill函数2.2.2 raise函数2.2.3 abort函数 2.3 通过异常的方式进行信号的产生2.4 通过软件条件的方式进行信号的产生2.4.1 关闭管道读端2.4.2 alarm函数 2.5 Core Dump&#x…

基于go语言的驾考系统设计与实现

在Internet时代&#xff0c;Internet信息技术已广泛应用于各个领域。 对人们的生活以及学习产生了较大的影响。通过信息技术建立的驾照考试管理系统&#xff0c;利用系统对驾照考试进行统一的管理&#xff0c;能够提驾照考试管理的工作效率&#xff0c;具有重要的现实意义。 本…

鸿蒙打包发布

HarmonyOS应用/元服务发布&#xff08;打包发布&#xff09; https://developer.huawei.com/consumer/cn/doc/harmonyos-guides-V13/ide-publish-app-V13?catalogVersionV13 密钥&#xff1a;包含非对称加密中使用的公钥和私钥&#xff0c;存储在密钥库文件中&#xff0c;格式…

基于Linux系统指令使用详细解析

一 Linux系统常用操作命令编辑快捷 1.1终端快捷键&#xff1a; Ctrl a/Home 切换到命令行开始 Ctrl e/End 切换到命令行末尾 Ctrl l 清除屏幕内容&#xff0c;效果等同于 clear Ctrl u 清除剪切光标之前的内容 Ctrl k 剪切清除光标之后的内容 Ctrl y 粘贴刚才所删…

深度学习-87-大模型训练之预训练和微调所用的数据样式

文章目录 1 大模型训练的阶段1.1 预训练1.1.1 全量预训练1.1.2 二次预训练1.2 微调2 预训练需要的数据2.1 清洗成的文本文档2.2 如何从文本文档学习2.3 常见预训练中文语料库3 微调需要的数据3.1 微调例子一:电商客服场景3.2 微调例子二:行政咨询场景3.3 微调数据长什么样3.3…

基于 STM32 的多功能时间管理器项目

引言 在快节奏的生活中&#xff0c;时间管理显得尤为重要。本项目旨在通过 STM32 开发一个多功能时间管理器&#xff0c;功能包括计时器、闹钟和日历。用户可以方便地设置不同的提醒和计时任务&#xff0c;以更好地管理日常生活和工作。 项目名称 多功能时间管理器 环境准备 …

麦田物语学习笔记:代码链接UI实现时间日期对应转换

基本流程 时间系统UI如下 本篇文章将UI和TimeManager里的数据联系在一起, 1.代码思路 (1)新建TimeUI.cs挂载在GameTime物体上,然后获取它的子物体这些组件来改变里面的数值,所以需要获得Day & Night的子物体Image中的Rect Transform,用于旋转季节的图标;获得Clock每个子物…