进程间通信

1、命名管道
(1)创建

int ret = mkfifo("/home/myfifo", 0766);
if (-1 == ret)
{
	perror("管道文件创建失败");
	return -1;
}

(2)写管道文件

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define SIZE 1024

int main()
{
	int fd = open("/home/myfifo", O_WRONLY);
	if (-1 == fd)
	{
		perror ("打开文件失败");
		return -1;
	}
	
	char buf[SIZE] = {0};
	while (1)
	{
		fgets(buf, SIZE, stdin);
		write(fd, buf, strlen(buf));
	}
	close (fd);
	return 0;
}

(3)读管道文件

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

#define SIZE 1024
int main()
{
	int fd = open("/home/myfifo", O_RDONLY);
	if (-1 == fd)
	{
		perror ("打开文件失败");
		return -1;
	}
	char buf[SIZE+1] = {0};
	while (1)
	{
		ssize_t ret = read(fd, buf, SIZE);
		if (-1 == ret)
		{
			perror("父进程读数据失败");
			break;
		}
		
		buf[ret] = '\0';
		printf ("读到数据:%s\n", buf);
		
	}
	close (fd);
	return 0;
}

(4)非阻塞读管道文件

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#define SIZE 1024
int main()
{
	int fd = open("/home/myfifo", O_RDONLY | O_NONBLOCK);
	if (-1 == fd)
	{
		perror ("打开文件失败");
		return -1;
	}
	char buf[SIZE+1] = {0};
	while (1)
	{
		ssize_t ret = read(fd, buf, SIZE);
		if (-1 == ret)
		{
			if (errno == EAGAIN)
			{
				printf("无数据可读\n");
				sleep(2);
				continue;
			}
			perror("父进程读数据失败");
			break;
		}
		
		buf[ret] = '\0';
		printf ("读到数据:%s\n", buf);
		
	}
	close (fd);
	return 0;
}

2、共享内存
(1)头文件

#ifndef _MSG_H_
#define _MSG_H_

typedef struct student
{
	char name[20];
	int id;
}Student;


#endif // _MSG_H_

(2)写入

#include <stdio.h>
#include <sys/ipc.h>
#include <sys/types.h>
#include <sys/shm.h>
#include "msg.h"
#include <string.h>
int main()
{
	// 1、创建或者获取操作共享内存的标识
	int shmid = shmget((key_t)1234, sizeof(Student), IPC_CREAT);
	if (-1 == shmid)
	{
		perror("共享内存创建失败");
		return -1;
	}
	
	// 2、将共享内存映射到本地,当做本地内存使用
	Student *ps = (Student*)shmat(shmid, NULL, 0);
	if (NULL == ps)
	{
		perror("共享内存映射失败");
		return -1;
	}
	
	strcpy(ps->name, "小明");
	ps->id = 10;
	
	// 3、解除映射
	shmdt(ps);

	
	// 4、删除共享内存
	shmctl(shmid, IPC_RMID, NULL);

	
	
	return 0;
}

(3)读取

#include <stdio.h>
#include <sys/ipc.h>
#include <sys/types.h>
#include <sys/shm.h>
#include "msg.h"
#include <string.h>
int main()
{
	// 1、创建或者获取操作共享内存的标识
	int shmid = shmget((key_t)1234, sizeof(Student), IPC_CREAT);
	if (-1 == shmid)
	{
		perror("共享内存创建失败");
		return -1;
	}
	
	// 2、将共享内存映射到本地,当做本地内存使用
	Student *ps = (Student*)shmat(shmid, NULL, 0);
	if (NULL == ps)
	{
		perror("共享内存映射失败");
		return -1;
	}
	
	printf ("name = %s, id = %d\n", ps->name, ps->id);
	
	// 3、解除映射
	shmdt(ps);
	
	// 4、删除共享内存
	
	return 0;
}

3、信号量
(1)头文件
ticket.h

#ifndef _TICKET_H_
#define _TICKET_H_

typedef struct shm
{
	int ticket;
}SHM;


#endif // _TICKET_H_

semaphore.h

#ifndef __SEMAPHORE_H__
#define __SEMAPHORE_H__
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <string.h>

union semun {
   int              val;    /* Value for SETVAL */
   struct semid_ds *buf;    /* Buffer for IPC_STAT, IPC_SET */
   unsigned short  *array;  /* Array for GETALL, SETALL */
   struct seminfo  *__buf;  /* Buffer for IPC_INFO(Linux specific) */
};

// 信号量初始化
int init_smph(int semid)
{
	union semun sem;
	sem.val = 1;
	
	// 给信号量 semid进行初始化
	int ret = semctl(semid, 0, SETVAL, sem);
	if(ret == -1)
	{
		perror ("semctl");
		return -1;
	}
	
	return 0;
}

// 信号量的P操作
int semaphore_p(int semid)
{
	struct sembuf sem;
	memset (&sem, 0, sizeof(sem));
	
	sem.sem_num = 0;
	sem.sem_op  = -1;
	sem.sem_flg = SEM_UNDO;
	int ret = semop(semid, &sem, 1);
	if (ret == -1)
	{
		perror ("semop_p");
		return -1;
	}
	return 0;
}

// 信号量的v操作
int semaphore_v(int semid)
{
	struct sembuf sem;
	memset (&sem, 0, sizeof(sem));
	
	sem.sem_num = 0;
	sem.sem_op  = 1;
	sem.sem_flg = SEM_UNDO;
	int ret = semop(semid, &sem, 1);
	if (ret == -1)
	{
		perror ("semop_v");
		return -1;
	}
	return 0;
}

// 信号量的删除
int del_smph(int semid)
{	
	int ret = semctl(semid, 0, IPC_RMID);
	if(ret == -1)
	{
		perror ("semctl");
		return -1;
	}
	
	return 0;
}

#endif // __SEMAPHORE_H__

(3)main.c

#include <stdio.h>
#include <sys/ipc.h>
#include <sys/types.h>
#include <sys/shm.h>
#include "ticket.h"
#include "semaphore.h"
#include <time.h>

void sellticket(int semid, SHM *ps)
{
	srand((unsigned int)time(NULL));
	
	while (1)
	{
		usleep(100000*(rand()%10+1));
		
		semaphore_p(semid);          // p操作
	
		if (0 == ps->ticket)
		{
			printf ("票卖完了\n");
			semaphore_v(semid);      // v操作
			break;
		}
		printf ("卖了一张票,座位号:%d\n", ps->ticket);
		ps->ticket -= 1;
	
		semaphore_v(semid);          // v操作
	}
}


int main(int argc, char **argv)
{
	int shmid = shmget((key_t)456, sizeof(SHM), IPC_CREAT);    // 创建或者获取共享内存
	if (-1 == shmid)
	{
		perror("共享内存创建失败");
		return -1;
	}
  
	SHM *ps = (SHM*)shmat(shmid, NULL, 0);                     // 共享内存映射
	if (NULL == ps)
	{
		perror("共享内存映射失败");
		return -1;
	}
	                       
	int semid = semget((key_t)789, 1, IPC_CREAT);              // 创建信号量
	if (-1 == semid)
	{
		perror("信号量创建失败");
		return -1;
	}
	
	if (2 == argc)            // 初始化
	{
		ps->ticket = 100;
		init_smph(semid);  
	}
	
	
	
	sellticket(semid, ps);    // 卖票
	
	sleep(2);
	if (2 == argc)
	{
		del_smph(semid);
		shmctl(shmid, IPC_RMID, NULL);
	}
	
	return 0;
}

4、信号
(1)信号捕捉

#include <stdio.h>
#include <signal.h>

void handle_signal(int sigNum)
{
	if (sigNum == 2)
		printf ("捕获到  Ctrl + c\n");
	else if (sigNum == 15)
		printf ("捕获到  kill 信号\n");
}

int main()
{	
	signal(SIGINT, handle_signal);    // Ctrl + c
	signal(SIGTERM, handle_signal);   // Ctrl + c
	while (1)
	{
		printf ("a\n");
		sleep(2);
	}

	return 0;
}

(2)定时器

#include <stdio.h>
#include <signal.h>
#include <unistd.h>

void handle_signal(int sigNum)
{
	if (sigNum == SIGALRM)
	{
		printf ("hello\n");
		alarm(2);      // 重置定时时间
	}
}

int main()
{	
	signal(SIGALRM, handle_signal);    // 定时信号
	alarm(2);          // 定时5秒
	
	while (1);

	return 0;
}

(3)处理子进程退出

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>

void handle_signal(int sigNum)
{
	pid_t pid;

	while ((pid = waitpid(-1, NULL, WNOHANG)) > 0)
	{
		printf ("处理了一个子进程退出信息,子进程的id = %d\n", pid);
	}
}

int main()
{	
	signal(SIGCHLD, handle_signal);    // 定时信号
	
	
	int i;
	for (i = 0; i < 10; i++)
	{
		pid_t pid = fork();
		if (0 == pid)
			break;
	}
	
	while (1);
	
	


	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值