// 、、使用内存映射可以拷贝文件
/*
对原始文件进行内存映射
创建一个新文件
把新文件的数据拷贝映射到内存中
通过内存拷贝将第一个文件的内存映射区拷贝到第二个文件的内存映射区
释放资源
*/
// 匿名映射,不需要文件实体来进行内存映射
// 只能用于有血缘关系的进程间通信
#include <sys/mman.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdlib.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
// int main()
// {
/*
// kill函数
// pid》0将信号发送给指定进程
// =0:将信号发送给当前进程组
// =-1:将信号发送给每一个有权限接受这个信号的进程
// 《-1:进程组id的绝对值
// sig,要发送的信号
// kill(getppid(), 9);
// kill(getpid(), 9);
// raise函数,给当前进程发送信号
// sig,要发送的信号,成功返回0,失败返回非0*/
// alarm函数
// 设置定时器,倒计时,当时间为0的时候,会给当前进程发送一个 SIGALARM信号
// 取消定时器,alarm(0)即可
// 返回值是unsigned int,倒计时剩余的时间(上一个alarm剩余的时间)
// SIGALRARM信号默认终止当前进程,每个进程只能设置一个定时器
// int seconds = alarm(5);
// printf("%d\n", seconds);
// sleep(2);
// seconds = alarm(3);
// printf("%d\n", seconds);
// while(1){
// }
// 1秒钟电脑能输多少个数?
//无论进程处于什么状态,alarm都会正常计时
// 实际时间=内核时间+用户时间+io消耗的时间
// alarm(1);
// int i =0;
// while(1){
// printf("%d\n",i++);
// }
// settimeer函数,用于替换alarm函数,且精度比alarm要高很多,
// 参数
// which,定时器以什么时间计时
// ITIME_REAL: 真实时间,时间到达,就发送SIGALRM
// ITIME_VIRTUAL;用户时间,时间到达就发送SIGVTALRM
// ITIMER_PROF:进程在内核状态下消耗的时间
// 过3秒后,每两秒执行一次
// struct itimerval new_value;
// // 设置值
// new_value.it_interval.tv_sec = 2;
// new_value.it_interval.tv_usec =0;
// new_value.it_value.tv_sec=3;
// new_value.it_value.tv_usec=0;
// int ret = setitimer(ITIMER_REAL,&new_value,NULL);
// printf("定时器开始\n");
// if (ret==-1){
// perror("settimer");
// exit(0);
// }
// getchar();
// signal捕获信号
// sigkill,sigstop既不能被捕捉也不能被忽略
// signum:要捕捉的信号
// handler:捕捉到信号之后如何处理
// sig_ign:忽略
// sig_dfl:使用信号默认的行为
// 回调函数:捕获到信号后执行该函数
// 过3秒后,每两秒执行一次
// 注册信号捕捉函数
//忽略信号
// signal(SIGALRM,SIG_IGN);
//直行信号的默认动作(sig_ALRM结束程序)
// signal(SIGALRM,SIG_DFL);
// void (*sig_handler)(int),这是一个函数指针。返回void,参数int,函数名称:sig_handler
void sig_handler(int num) {
printf("捕获到了信号:%d\n", num);
printf("lllllllllllllll]n");
}
int main()
{
// 必须在设置定时器之前注册回调函数
sig_t retr = signal(SIGALRM,sig_handler);
if(retr==SIG_ERR) {
perror("signal注册");
exit(-1);
}
struct itimerval new_value;
// 设置值
new_value.it_interval.tv_sec = 2;
new_value.it_interval.tv_usec =0;
new_value.it_value.tv_sec=3;
new_value.it_value.tv_usec=0;
int ret = setitimer(ITIMER_REAL,&new_value,NULL);
printf("定时器开始\n");
if (ret==-1){
perror("settimer");
exit(0);
}
getchar();
// pid_t pid = fork();
// if(pid==0){
// for(int i=0;i<5;i++) {
// printf("孩子进程%d\n", i);
// sleep(1);
// }
// }else if(pid>0) {
// printf("爹进程\n");
// sleep(2);
// printf("傻儿子\n");
// kill(pid,SIGINT);
// }
return 0;}
/*
// char * buf;
// strcpy(buf, "cao nima");return 0;}/*
// 修改mmap的flags参数,之前使用的是MAP_SHARED
// 现在我们要用的是MAP_ANON
int len = 1204;
void *ptr = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_ANON | MAP_SHARED, -1, 0);
if (ptr == MAP_FAILED)
{
perror("mmap");
exit(0);
}
// 父子进程间通信
int pid = fork();
if (pid > 0)
{
strcpy((char *)ptr, "hello world");
wait(NULL);
}
else if (0 == pid)
{
sleep(1);
printf("%s\n", (char *)ptr);
}
// 释放内存映射区
int ret = munmap(ptr, len);
// int fd = open("englsih.txt", O_RDONLY);
// // 获取原始文件的大小
// int len = lseek(fd, 0, SEEK_END);
// // 创建一个新文件,并对其大小进行拓展
// int fd1 = open("cpy.txt", O_CREAT | O_RDWR, 0664);
// truncate("cpy.txt", len);
// write(fd1, " ", 1);
// // 分别进行内存映射
// void *ptr = mmap(NULL, len, PROT_READ, MAP_SHARED, fd, 0);
// void *ptr1 = mmap(NULL, len, PROT_READ | PROT_WRITE, MAP_SHARED, fd1, 0);
// if (ptr == MAP_FAILED)
// {
// perror("mmap");
// exit(0);
// }
// if (ptr1 == MAP_FAILED)
// {
// perror("mmap");
// exit(0);
// }
// // 内存拷贝
// memcpy(ptr1, ptr, len);
// munmap(ptr1, len);
// munmap(ptr, len);
// close(fd1);
// close(fd);
return 0;
}
*/
signal函数对的返回值是
从定义中可以看到,不同的linux操作系统需要使用不同的标识符
在本例中,我是用的是wsl ubuntu,使用sighandler_t
就会提示无法找到该标识符,使用sig_t
就可以正常变异了