3个进程之间的相互通信linux,进程间通信--管道

1.2有名管道的创建

该函数的第一个参数是一个普通的路劲名,也就是创建后FIFO的名字。第二个参数与打开普通文件的open()函数中的mode参数相同。如果mkfifo的一个参数是一个已经存在路劲名时,会返回EEXIST错误,所以一般典型的调用代码首先会检查是否返回该错误,如果确实返回该错误,那么只要调用打开FIFO的函数就可以了。

1.3有名管道的打开规则

有名管道比无名管道多了一个打开操作:open

FIFO的打开规则:

如果当前打开操作时为读而打开FIFO时,若已经有相应进程为写而打开该FIFO,则当前打开操作将成功返回;否则,可能阻塞到有相应进程为写而打开该FIFO(当前打开操作设置了阻塞标志);或者,成功返回(当前打开操作没有设置阻塞标志)。

如果当前打开操作时为写而打开FIFO时,如果已经有相应进程为读而打开该FIFO,则当前打开操作将成功返回;否则,可能阻塞直到有相应进程为读而打开该FIFO(当前打开操作设置了阻塞标志);或者,返回ENIO错误(当期打开操作没有设置阻塞标志)。

案例:

A.open for  write

点击(此处)折叠或打开

#include

#include

#include

#include

#include

#include

#include

int main(int argc,char *argv[])

{

int fd;

if(argc < 2)

{

fprintf(stderr,"usage : %s argv[1].\n",argv[0]);

exit(EXIT_FAILURE);

}

if(mkfifo(argv[1],0666) < 0 && errno != EEXIST)

{

fprintf(stderr,"Fail to mkfifo %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

if((fd = open(argv[1],O_WRONLY)) < 0)

{

fprintf(stderr,"Fail to open %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

printf("open for write success.\n");

return 0;

}

B.open for read

点击(此处)折叠或打开

#include

#include

#include

#include

#include

#include

#include

int main(int argc,char *argv[])

{

int fd;

if(argc < 2)

{

fprintf(stderr,"usage : %s argv[1].\n",argv[0]);

exit(EXIT_FAILURE);

}

if(mkfifo(argv[1],0666) < 0 && errno != EEXIST)

{

fprintf(stderr,"Fail to mkfifo %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

if((fd = open(argv[1],O_RDONLY)) < 0)

{

fprintf(stderr,"Fail to open %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

printf("open for read success.\n");

return 0;

}

探究发现,如果open时没有使用O_NONBLOCK参数,我们发现不论读端还是写端先打开,先打开者都会阻塞,一直阻塞到另一端打开。

读者自己可以探究,如果open时使用了O_NONBLOCK参数,此时打开FIFO 又会是什么情况?

1.4有名管道的读写规则

A.从FIFO中读取数据

约定:如果一个进程为了从FIFO中读取数据而以阻塞的方式打开FIFO, 则称内核为该进程的读操作设置了阻塞标志

<1>如果有进程为写而打开FIFO,且当前FIFO内没有数据,则对于设置了阻塞标志的读操作来说,将一直阻塞。对于没有设置阻塞标志读操作来说返回-1,当前errno值为EAGAIN,提醒以后再试。

<2>对于设置阻塞标志的读操作说,造成阻塞的原因有两种:当前FIFO内有数据,但有其他进程正在读这些数据;另外就是FIFO内没有数据。解阻塞的原因则是FIFO中有新的数据写入,不论写入数据量的大小,也不论读操作请求多少数据量。

<3>如果没有进程写打开FIFO,则设置了阻塞标志的读操作会阻塞

<4>如果写端关闭,管道中有数据读取管道中的数据,如果管道中没有数据读端将不会继续阻塞,此时返回0。

注意:如果FIFO中有数据,则设置了阻塞标志的读操作不会因为FIFO中的字节数小于请求读的字节数而阻塞,此时,读操作会返回FIFO中现有的数据量。

B.向FIFO中写入数据

约定:如果一个进程为了向FIFO中写入数据而阻塞打开FIFO,那么称该进程内的写操作设置了阻塞标志。

对于设置了阻塞标志的写操作:

<1>当要写入的数据量不大于PIPE_BUF时,linux将保证写入的原子性。如果此时管道空闲缓冲区不足以容纳要写入的字节数,则进入睡眠,直到当缓冲区中能够容纳写入的字节数时,才开始进行一次性写操作。

<2>当要写入的数据量大于PIPE_BUF时,Linux将不再保证写入的原子性。FIFO缓冲区一有空闲区域,写进程就会试图向管道写入数据,写操作在写完所有请求写的数据后返回。

对于没有设置阻塞标志的写操作:

<1>当要写入的数据量大于PIPE_BUF时,linux将不再保证写入的原子性。在写满所有FIFO空闲缓冲区后,写操作返回。

<2>当要写入的数据量不大于PIPE_BUF时,linux将保证写入的原子性。如果当前FIFO空闲缓冲区能够容纳请求写入的字节数,写完后成功返回;如果当前FIFO空闲缓冲区不能够容纳请求写入的字节数,则返回EAGAIN错误,提醒以后再写。

注意:只有读端存在,写端才有意义。如果读端不在,写端向FIFO写数据,内核将向对应的进程发送SIGPIPE信号(默认终止进程);

案例一、

write to FIFO

#include

#include

#include

#include

#include

#include

#include

#define MAX 655360

int main(int argc,char *argv[])

{

int n,fd;

char buf[MAX];

if(argc < 2)

{

fprintf(stderr,"usage : %s argv[1].\n",argv[0]);

exit(EXIT_FAILURE);

}

if(mkfifo(argv[1],0666) < 0 && errno != EEXIST)

{

fprintf(stderr,"Fail to mkfifo %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

if((fd = open(argv[1],O_WRONLY )) < 0)

{

fprintf(stderr,"Fail to open %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

printf("open for write success.\n");

while(1)

{

printf(">");

scanf("%d",&n);

n = write(fd,buf,n);

printf("write %d bytes.\n",n);

}

exit(EXIT_SUCCESS);

}

read from FIFO

点击(此处)折叠或打开

#include

#include

#include

#include

#include

#include

#include

#define MAX 655360

int main(int argc,char *argv[])

{

int fd,n;

char buf[MAX];

if(argc < 2)

{

fprintf(stderr,"usage : %s argv[1].\n",argv[0]);

exit(EXIT_FAILURE);

}

if(mkfifo(argv[1],0666) < 0 && errno != EEXIST)

{

fprintf(stderr,"Fail to mkfifo %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

if((fd = open(argv[1],O_RDONLY )) < 0)

{

fprintf(stderr,"Fail to open %s : %s.\n",argv[1],strerror(errno));

exit(EXIT_FAILURE);

}

printf("open for read success.\n");

while(1)

{

printf(">");

scanf("%d",&n);

n = read(fd,buf,n);

printf("Read %d bytes.\n",n);

}

exit(EXIT_SUCCESS);

}

读者可以将这两个程序运行,然后输入read和write   FIFO大小就可以看到效果。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值