linux-sys_进程间通信(IPC)

## pipe管道  (单向通信)

#include <unistd.h>

int pipe(int pipefd[2]);


#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>

int main(void)
{
	int fd[2];
	char str[1024] = "hello itcast";
	char buf[1024];
	pid_t pid;
	
	if(pipe(fd) < 0)
	{
		perror("pipe");
		exit(1);
	}
	
	pid = fork();
	//父写子读
	if (pid >0)
	{
		//父进程里,关闭父读
		close(fd[0]);
		write(fd[1],str,strlen(str));
		wait(NULL);
	}
	else if (pid == 0)
	{
		int len;
		close(fd[1]);
		len = read(fd[0],buf,sizeof(buf));
		write(STDOUT_FILENO,buf,len);
	}
	else
	{
		perror("fork");
		exit(1);
	}
	return 0;
}
使用管道需要注意以下4种特殊情况(假设都是阻塞I/O操作,没有设置O_NONBLOCK标
志):
1.如果所有指向管道写端的文件描述符都关闭了(管道写端的引用计数等于0),而仍
然有进程从管道的读端读数据,那么管道中剩余的数据都被读取后,再次read会返回0,就
像读到文件末尾一样。
2.如果有指向管道写端的文件描述符没关闭(管道写端的引用计数大于0),而持有管
道写端的进程也没有向管道中写数据,这时有进程从管道读端读数据,那么管道中剩余的数
据都被读取后,再次read会阻塞,直到管道中有数据可读了才读取数据并返回。
3.如果所有指向管道读端的文件描述符都关闭了(管道读端的引用计数等于0),这时
有进程向管道的写端write,那么该进程会收到信号SIGPIPE,通常会导致进程异常终止。讲
信号时会讲到怎样使SIGPIPE信号不终止进程。
4.如果有指向管道读端的文件描述符没关闭(管道读端的引用计数大于0),而持有管
道读端的进程也没有从管道中读数据,这时有进程向管道写端写数据,那么在管道被写满时
再次write会阻塞,直到管道中有空位置了才写入数据并返回。

## fifo  有名管道

创建一个有名管道,解决无血缘关系的进程通信


## mmap/munmap

mmap可以把磁盘文件的一部分直接映射到内存,这样文件中的位置直接就有对应的内存
地址,对文件的读写可以直接用指针来做而不需要read/write函数。


#include <sys/mman.h>
void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);
int munmap(void *addr, size_t length);


如果addr参数为NULL,内核会自己在进程地址空间中选择合适的地址建立映射。如果
addr不是NULL,则给内核一个提示,应该从什么地址开始映射,内核会选择addr之上的某个
合适的地址开始映射。建立映射后,真正的映射首地址通过返回值可以得到。len参数是需
要映射的那一部分文件的长度。off参数是从文件的什么位置开始映射,必须是页大小的整
数倍(在32位体系统结构上通常是4K)。filedes是代表该文件的描述符。
prot参数有四种取值:

* PROT_EXEC表示映射的这一段可执行,例如映射共享库
* PROT_READ表示映射的这一段可读
* PROT_WRITE表示映射的这一段可写
* PROT_NONE表示映射的这一段不可访问
flag参数有很多种取值,这里只讲两种,其它取值可查看mmap(2)
* MAP_SHARED多个进程对同一个文件的映射是共享的,一个进程对映射的内存做了修
改,另一个进程也会看到这种变化。
* MAP_PRIVATE多个进程对同一个文件的映射不是共享的,一个进程对映射的内存做了修
改,另一个进程并不会看到这种变化,也不会真的写到文件中去。
如果mmap成功则返回映射首地址,如果出错则返回常数MAP_FAILED。当进程终止时,该进程
的映射内存会自动解除,也可以调用munmap解除映射。munmap成功返回0,出错返回-1。


**用于进程间通信时,一般设计成结构体,来传输通信的数据

**进程间通信的文件,应该设计成临时文件

**当报总线错误时,优先查看共享文件是否有存储空间


/* procee_mmap_w.c */

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>

#define MAPLEN 0x1000

struct STU
{
	int id;
	char name[20];
	char sex;
};

void sys_err(char *str, int exitno)
{
	perror(str);
	exit(exitno);
}

int main(int argc, char *argv[])
{
	struct STU *mm;
	int fd, i = 0;
	if(argc < 2)
	{
		printf("./a.out filename\n");
		exit(1);
	}
	fd = open(argv[1], O_RDWR | O_CREAT, 0777);
	if(fd < 0)
		sys_err("open", 1);

	if (lseek(fd, MAPLEN-1, SEEK_SET) < 0)
		sys_err("lseek", 3);

	if(write(fd, "\0", 1) < 0)
		sys_err("write", 4);

	mm = mmap(NULL, MAPLEN, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
	if (mm == MAP_FAILED)
		sys_err("mmap", 2);
	
	close(fd);	

	while (1)
	{
		mm->id = i;
		sprintf(mm->name, "zhhg-%d", i);
		if(i % 2 == 0)
			mm->sex = 'm';
		else
			mm->sex = 'w';
		i++;
		sleep(1);
	}
	munmap(mm, MAPLEN);
	return 0;
}

/* procee_mmap_r.c */

#include <stdio.h>#include <sys/types.h>#include <sys/stat.h>#include <sys/mman.h>#include <unistd.h>#include <stdlib.h>#include <fcntl.h>#define MAPLEN 0x1000struct STU{int id;char name[20];char sex;};void sys_err(char *str, int exitno){perror(str);exit(exitno);}int main(int argc, char *argv[]){struct STU *mm;int fd, i = 0;if(argc < 2){printf("./a.out filename\n");exit(1);}fd = open(argv[1], O_RDWR);if(fd < 0)sys_err("open", 1);mm = mmap(NULL, MAPLEN, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);if (mm == MAP_FAILED)sys_err("mmap", 2);close(fd);unlink(argv[1]);while (1){printf("%d\n", mm->id);printf("%s\n", mm->name);printf("%c\n", mm->sex);sleep(1);}munmap(mm, MAPLEN);return 0;}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值