进程信技术简称IPC,可以利用此技木让多个进程相传建消数据,有大量的进程间通信方案
进程有虚拟地址空间,进程间通信在虚拟地址的哪一层实现的?
绝大多数进程间通信是在内核层完成的, 因为内核层内存共享
为了便于开发者使用管道可以使用pipe函数创建管道,创建成功后会生成两个文件描述符,分别指向管道的读端fds[0]和写端fds[1]。匿名管道只能再有亲缘关系的进程之间使用
两个进程公用一个管道,实现通信
匿名管道只能在有亲缘关系的进程间的通信
pipe(int fds[2]);
注意:管道创建要在 创建子进程之前完成, 避免子进程创建管道
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define MSG "HI"
int main()
{
pid_t pid;
int fds[2];
pipe(fds);
pid = fork();
// 父进程写,子进程读
if (pid > 0)
{
close(fds[0]); // 父进程关闭读功能,文件描述符-1
printf("parent pid %d alive..\n", getpid());
sleep(5);
write(fds[1], MSG, strlen(MSG)); // 父进程写入信息
printf("parent pid %d send msg success..\n", getpid());
close(fds[1]); // 父进程使用完成写的功能,关闭写功能,文件描述符-1
}
else if (pid == 0)
{
close(fds[1]); // 子进程关闭写功能,文件描述符-1
char buffer[1024];
bzero(buffer, sizeof(buffer));
read(fds[0], buffer, sizeof(buffer)); // 子进程读管道中的内容
printf("child read msg %s \n", buffer);
close(fds[0]); // 子进程使用完成读的功能,关闭写功能,文件描述符-
exit(0);
}
else
{
perror("FORK CALL FAILED");
exit(0);
}
}
工作方式
单工 : 确定通信方向, 非读即写, 不可变更
半双工/可调节单工: 非读即写,但是不同时刻可以调节读写
双工:可以同时读写
这四种情况具有普遍意义
在一个C/S模型中,客户端发送了一条请求消息之后客户端异常关闭,服务端回复结果时异常关闭,服务端异常关闭原因?
(SIGPIPE信号)send(sockfd , buffer, size , MSG NOSIGNAL)
是一个内核缓冲区,为环形队列结构, 大小为4k,命名管道可以解决无关联进程间的通信
管道通信与传统的文件传输不同,读取文件内容并不会除文件内容,但是管道是队列结构,出队的数据会从管道消失
可以持久化存储的磁盘文件并不适合作为传输介质
简单理解,管道文件是一个指向内核管道缓冲区的指针,所有向管道文件读写的操作,都会重定向到内核管道中
mkfifo pname #命令创建管道文件
mkfifo(char * pname,0664) #函数创建管道文件
管道文件的访问是要凑齐两种权限的 即使 RDWR,如果某个进程以一种权限打开管道文件, open函数会立即阻塞,等待另一种权限。即使只有一个进程 只要满足读写权限,就可以成功打开管道了
一个读端中存在多个读序列,阻塞读只对第一个读序列有效,其他读序列被设置为非阻塞
判定是原子传输或是非原子,取决于用户发送数据包大小,如果数据包<=管道大小, 则是原子传输,如果数据包>管道大小(即4K) 就是非原子传输
非原子访问,最大化的利用容器空间, 只有容器满时才会限制写端,读写效率传输效率高,但是数据会被多次拆分,会对读端造成一些麻烦,读端要验证数据包完整性后才可以使用
是一种内存映射文件的技术,它允许将文件映射到进程的虚拟内存空间,使得文件的内容可以像访问内存一样被读取和写入。
#include
void * ptr = mmap(NULL , size , PROT READIPROT WRITE , int how , int fd , 0);
共享映射 MAP_SHARED,共享映射有 Sync同步机制
私有映射 MAP_PRIVATE
多进程间利用 MAP_SHARED的Sync同步机制可以实现共享映射实现进程通信
成功返回映射内存地址ptr
失败返回MAP_FAILED关键字
munmap(ptr,size); //释放映射内存
ftruncate(fd,sizeof(msg_t)*4096)
#include
#include
#include
#include
#include
#include
#include
#include
int main()
{
int fd;
fd = open("map_file", O_RDWR);
int size = lseek(fd, 0, SEEK_END);
int *ptr = NULL;
if ((ptr = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0)) == MAP_FAILED)
{
perror("mmap call failed");
exit(0);
}
close(fd);
sleep(10);
ptr[0] = 0x34333231;
munmap(ptr, size);
return 0;
}
最后一个参数,控制偏移量
映射偏移量必须是4k的整数倍,因为内存分页一页为4k。
零拷贝并不是指完全节省拷贝或切换开销,而是尽可能较少