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;
}