12_有名无名管道、dup2()
12_有名无名管道、dup2()
1.进程间通信IPC
管道 信号量 共享内存 消息队列 套接字
ps -ef | grep “bash” //查找ps -ef,并放入管道,grep查找关键字”bash”
2.管道
①必须两个进程同时打开,一读一写
②打开管道open 阻塞
③写入数据write 管道满了就会阻塞写 内存
④读管道read 空了就会阻塞读 内存
⑤close
2.1有名管道
//创建管道
//a.c 写入管道
//b.c 读取管道
//终端效果
其中,如果写端a.c ^C提前结束,读端b.c读到n=0,也退出;
如果读端^C提前结束,写端收到异常信号SIGPIPE,下一个put后结束。
想看信号的话,在写端增加signal。signal(SIGPIPE,fun),fun中打印sig即可。
2.2无名管道
pipe(int fd[]) fd[0]读端 fd[1]写端
父进程用来写,要先关闭fd[0];子进程用来读,要先关闭fd[1]。(子进程和父进程使用的同一个管道)
父进程关闭读端只能写,然后输入 写 输入 写…; 子进程关闭写端只能读,然后读 不为空 打印 读 不为空 打印…。
1.有名和无名的区别? 有名可以在任意两个进程间通信,无名只能在父子进程
2.写入管道的数据在哪里?内存?磁盘? 内存
3.管道的通信方式是全双工还是半双工? 半双工 (全双工:既能发送也能收听,可同时发生。 半双工:对讲机,能发送也能收听,但不能同时发生。 单工:收音机。)
3.dup()
dup2()
dup2(3,1)
3覆盖1,1原本的标准输出现在指向file.txt,(由于1位置标准输出printf最终会调用系统调用write(1),若覆盖1,则write到了file.txt里),相当于把输出内容写入到了file.txt文件中。
3.1dup2() 例1
行号 | 注释 |
---|---|
10 | fd=3处创建了一个文件file.txt |
13 | 3号处覆盖1号处,1号处printf最终调用的系统调用write(1),其实写入了3号的文件file.txt中,主函数运行结果不会打印,打印内容存在了file.txt中 |
3.2dup2() 例2
(如果我们用管道的写端覆盖掉1号处的标准输出,那么标准输出print - wirte(1)将会写到管道。)
实现:ps -ef | grep bash
思想:先复制一个子进程,然后子进程的写端覆盖掉标准输出,子进程再替换成另一个程序(ps -ef),这个程序的标准输出的打印即写入了管道,然后我们在父进程中读管道,即读出了ps -ef的输出。相当于父进程读出子进程的结果。
行号 | 注释 |
---|---|
33 | 父进程先关闭写端,保证写端由子进程控制 |
42 | 最后,父进程记得关闭读端 |