c/c++ Linux:信号(1):signal函数、pause函数、alarm函数
liebian365 2024-10-30 04:48 21 浏览 0 评论
更多linux知识:linux目录索引
一、信号
1.什么是信号?2.哪些情况下会产生信号?3.如何查看信号?4.常用信号解释5.处理的三种方式二、操作信号
1.注册信号
①作用②注册信号所使用的函数signal()③实例验证④另一个可以获取信号的函数pause()2.给进程发送信号
①命令方式②函数方式③实例验证④补充另外两个可以发送消息的函数三、SIGALRM信号
1.alarm函数2.实例验证3.alarm的超时处理:简单的考试计时程序4.alarm的定时处理
①用于定时的函数:setitimer()②对struct itimerval *new_value的详解③定时处理的实例:
一、信号
1.什么是信号?
系统为了响应某些状况而产生的事件。进程收到信号后采取相应的动作。
2.哪些情况下会产生信号?
①键盘事件,如:ctl c 、ctl \ ②访问非法内存 ③硬件出现故障 ④用户态到内核态的切换
3.如何查看信号?
指令:kill -l 查看系统所有信号
4.常用信号解释
5.处理的三种方式
当进程接收到信号时,有三种方式处理该信号 ①忽略:收到信号后,不处理 SIGKILL和SIGSTOP是不可能被忽略的 ②捕获并处理:收到信号后,执行我们自己写的代码 SIGKILL和SIGSTOP不能捕获 ③默认处理方式
二、操作信号
1.注册信号
①作用
注册信号实际是对信号进行三种处理操作,用于告诉当前进程接收信号后该去执行什么动作
②注册信号所使用的函数signal()
1.函数原型: typedef void (*sighandler_t)(int); sighandler_t signal(int signum, sighandler_t handler); 2.函数功能解释: 开始获取信号值为signum的信号,如果获取到该信号,则开始执行handler指向的函数 3.返回值: 成功返回原本的信号处理函数指针,失败返回 SIGERR, SIGERR的宏为 #define SIG_IGN ((sighandler_t)-1) 4.参数解释 signum:指明了所要处理的信号类型,它可以取除了SIGKILL和SIGSTOP外的任何一种信号。 handler:用户自定义的函数;第二个参数不仅可以传函数指针handler,它还可以取以下三种值,如下表:
③实例验证
//验证SIG_IGN #include<stdio.h> #include<unistd.h> #include<signal.h> #include<stdlib.h> int main() { int i=0; signal(SIGINT,SIG_IGN);//SIGINT是ctl c发起的信号 //SIG_IGN表示忽略接收到的信号 for(i=0;i<10;++i)//执行10秒 { printf("死不了:%d\n",i); sleep(1); } return 0; }
//验证SIG_DFL #include<stdio.h> #include<unistd.h> #include<signal.h> #include<stdlib.h> int main() { int i=0; signal(SIGINT,SIG_DFL);//SIGINT是ctl c发起的信号 //SIG_DFL表示采用默认处理方式处理接收到的信号 //把这句代码去掉,执行效果一样 for(i=0;i<10;++i)//执行10秒 { printf("死不了:%d\n",i); sleep(1); } return 0; }
//验证sighandler_t #include<stdio.h> #include<unistd.h> #include<signal.h> #include<stdlib.h> void handler(int n)//自定义的函数 { printf("死掉了\n"); exit(1); } int main() { signal(SIGINT,handler);//SIGINT是ctl c发起的信号 while(1) { printf("死不了\n"); sleep(1); } return 0; }
④另一个可以获取信号的函数pause()
1.函数原型: int pause(void); 2.功能解释: 暂停进程,把当前进程置成就绪态,让出CPU,直到收到任意一个信号后终止,并且当处理完该信号之后,直接执行pause()函数下面的语句 3.返回值: 只返回-1 4.参数解释: 不需要传参,如:pause();printf("pause over\n");//当前进程暂停,直到系统任意发出一个信号,pause()才被终止(打断),开始继续执行printf()。
2.给进程发送信号
①命令方式
指令:kill -信号值 pid
②函数方式
1.函数原型: int kill(int pid,int signum); 2.函数功能解释: 给进程id为pid的进程发送一个信号值为signum的信号 3.返回值: 成功返回0,失败返回-1 4.参数解释 signum:信号值,即信号编号 pid:进程id,它可以取以下四种值,如下表:
pid > 0 将信号发送给pid对应的进程
pid = 0 将信号发送给调用者所在进程组的所有进程
pid = -1 将信号发送给所有进程(1号进程除外)
pid < -1 将信号发送给进程组 id 为|pid|的进程组中所有进程
补充:进程组:如管道连接的进程、fork创建的父子进程
③实例验证
#include<stdio.h> #include<unistd.h> #include<signal.h> #include<stdlib.h> void handler(int n)//自定义函数,用于验证能接收到信号 { printf("我收到信号了\n"); } int main() { int i=0; signal(SIGUSR1,handler);//父进程在获取信号SIGUSR1 //SIGUSER1:用户自定义信号,常用于发送和接收 pid_t pid=fork(); if(pid == 0)//子进程 { sleep(1); kill(getppid(),SIGUSR1);//将信号SIGUSER1发送给父进程 exit(0); } else { while(1) { printf("滴滴\n"); sleep(3); } } return 0; }
④补充另外两个可以发送消息的函数
1.int rasie (int signum); 给自己发送信号 返回值:成功返回0;失败:返回-1 2.int killpg(int gid,int signum); 给进程组发送信号 返回值:-1,并把error值设为EINTR
三、SIGALRM信号
1.alarm函数
①函数原型: unsigned int alarm(unsigned int seconds); ②函数功能解释: 若干秒后,给当前进程发送一个SIGALRM信号 ③返回值: 成功:如果调用此alarm()前,进程已经设置了闹钟时间,则返回上一个闹钟时间的剩余时间,否则返回0。 失败:-1 ④参数解释 second > 0:当seconds秒后,触发SIGALRM信号 seconds = 0: 表示清除SIGALRM信号
2.实例验证
#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <signal.h> void handler(int s) { printf("超时\n"); exit(1); } int main( void ) { char buf[100] = { }; printf("输入名字:"); signal(SIGALRM, handler);//捕获到SIGALRM信号后执行handler函数 alarm(3);//设置倒计时为3 //如果3秒结束没有清除SIGALRM信号,则获取SIGALRM信号 scanf("%s", buf); alarm(0); //清楚SIGALRM信号 printf("名字为:%s\n", buf); for ( ; ; ) //用于验证alarm(0)确实清除了SIGALRM信号 { printf("滴滴\n"); sleep(1); } }
3.alarm的超时处理:简单的考试计时程序
#include <stdio.h> #include <stdlib.h> #include <signal.h> #include <unistd.h> int Right = 0;//做对的题数 int Wrong = 0;//做错的题数 void handler(int s) //用于验证超时 { printf("时间到\n"); printf("Right=%d, Wrong=%d\n", Right, Wrong); exit(1); } int main( void ) { int i=0; signal(SIGALRM, handler);//获取SIGALRM信号,执行handler函数 alarm(20);//设置倒计时为20 srand(getpid());//随机数种子 for (i=0; i<10; i++) //出10道10以内的加法题 { int left = rand()%10; int right = rand()%10; printf("%d+%d=", left, right); int ret; scanf("%d", &ret); if ( left + right == ret ) { Right++; } else { Wrong++; } } printf("做完了\n"); printf("Right=%d, Wrong=%d\n", Right, Wrong); }
4.alarm的定时处理
①用于定时的函数:setitimer()
1.函数原型: #include <sys/time.h> int setitimer(int which, const struct itimerval *new_value, struct itimerval *old_value); 2.函数功能解释: 设置一个定时器,从第n秒(n是自己设定)开始启动定时器,然后每隔m秒(m也是自己设定)发出一个SIGALRM信号 3.返回值: 返回-1表示失败,0表示成功 4.参数解释 which:一般写成ITIMER_REAL,即真实的桌面时间 new_value:一个结构体指针,结构体内部存储着“启动时间”和“间隔时间”,具体的在下面有详解 old_value:同样的结构体指针,一般写成NULL
②对struct itimerval *new_value的详解
1.结构体内部的信息 struct itimerval { struct timeval it_interval; /*以后每一次执行的间隔时间*/ struct timeval it_value; /*第一次执行的时间*/ }; struct timeval //时间的精确度 { long tv_sec; /* 时间的秒部分*/ long tv_usec; /*时间的微秒部分*/ }; 2.实例说明该结构体的作用 struct itimerval it; //上句代码的作用是定义一个该结构体变量it,it包括两部分信息:启动时间和间隔时间 it.it_value.tv_sec = 0; it.it_value.tv_usec = 1; //上两句代码的作用是将it的启动时间设为:真实的桌面时间+0秒1微秒 it.it_interval.tv_sec = 1; it.it_interval.tv_usec = 0; //上两句代码的作用是将it的间隔时间设为:1秒0微妙
③定时处理的实例:
#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <signal.h> #include <sys/time.h> void handler(int s) { printf("收到SIGALRM信号了\n"); } int main( void ) { signal(SIGALRM, handler);//如果获取到SIGALRM信号,则执行handler函数 struct itimerval it; //上句代码的作用是定义一个该结构体变量it,it包括两部分信息:启动时间和间隔时间 it.it_value.tv_sec = 0; it.it_value.tv_usec = 1; //上两句代码的作用是将it的启动时间设为:真实的桌面时间+0秒1微秒 it.it_interval.tv_sec = 1; it.it_interval.tv_usec = 0; //上两句代码的作用是将it的间隔时间设为:1秒0微妙 setitimer(ITIMER_REAL, &it, NULL); //上面的定时器内部信息设置完后,通过setitimer函数启动该定时器 while(1)//无限循环,用于验证定时器确实是每个一段时间就发出一个SIGALRM信号 ; }
更多linux免费视频资料获取 后台私信【架构】
相关推荐
- 精品博文嵌入式6410中蓝牙的使用
-
BluetoothUSB适配器拥有一个BluetoothCSR芯片组,并使用USB传输器来传输HCI数据分组。因此,LinuxUSB层、BlueZUSB传输器驱动程序以及B...
- win10跟这台计算机连接的前一个usb设备工作不正常怎么办?
-
前几天小编闲来无事就跑到网站底下查看粉丝朋友给小编我留言询问的问题,还真的就给小编看到一个问题,那就是win10跟这台计算机连接的一个usb设备运行不正常怎么办,其实这个问题的解决方法时十分简单的,接...
- 制作成本上千元的键盘,厉害在哪?
-
这是稚晖君亲自写的开源资料!下方超长超详细教程预警!!全文导航:项目简介、项目原理说明、硬件说明、软件说明项目简介瀚文智能键盘是一把我为自己设计的——多功能、模块化机械键盘。键盘使用模块化设计。左侧的...
- E-Marker芯片,USB数据线的“性能中枢”?
-
根据线缆行业的研究数据,在2019年搭载Type-C接口的设备出货量已达到20亿台,其中80%的笔记本电脑和台式电脑采用Type-C接口,50%的智能手机和平板电脑也使用Type-C接口。我们都知道,...
- ZQWL-USBCANFD二次开发通讯协议V1.04
-
修订历史:1.功能介绍1.1型号说明本文档适用以下型号: ZQWL-CAN(FD)系列产品,USB通讯采用CDC类实现,可以在PC机上虚拟出一个串口,串口参数N,8,1格式,波特率可以根据需要设置(...
- win10系统无法识别usb设备怎么办(win10不能识别usb)
-
从驱动入手,那么win10系统无法识别usb设备怎么办呢?今天就为大家分享win10系统无法识别usb设备的解决方法。1、右键选择设备管理器,如图: 2、点击更新驱动程序,如图: 3、选择浏览...
- 微软七月Win8.1可选补丁有内涵,含大量修复
-
IT之家(www.ithome.com):微软七月Win8.1可选补丁有内涵,含大量修复昨日,微软如期为Win7、Win8.1发布7月份安全更新,累计为6枚安全补丁,分别修复总计29枚安全漏洞,其中2...
- 如何从零开始做一个 USB 键盘?(怎么制作usb)
-
分两种情况:1、做一个真正的USB键盘,这种设计基本上不涉及大量的软件编码。2、做一个模拟的USB键盘,实际上可以没有按键功能,这种的需要考虑大量的软件编码,实际上是一个单片机。第一种设计:买现成的U...
- 电脑识别U盘失败?5个实用小技巧,让你轻松搞定USB识别难题
-
电脑识别U盘失败?5个实用小技巧,让你轻松搞定USB识别难题注意:有些方法会清除USB设备里的数据,请谨慎操作,如果不想丢失数据,可以先连接到其他电脑,看能否将数据复制出来,或者用一些数据恢复软件去扫...
- 未知usb设备设备描述符请求失败怎么解决
-
出现未知daousb设备设备描述符请求失du败解决办zhi法如下:1、按下Windows+R打开【运行】;2、在版本运行的权限输入框中输入:services.msc按下回车键打开【服务】;2、在服务...
- 读《飘》47章20(飘每章概括)
-
AndAhwouldn'tleaveMissEllen'sgrandchildrenfornotrashystep-patobringup,never.Here,Ah...
- 英翻中 消失的过去 37(消失的英文怎么说?)
-
翻译(三十七):消失的过去/茱迪o皮考特VanishingActs/JodiPicoult”我能做什么?“直到听到了狄利亚轻柔的声音,我才意识到她已经在厨房里站了好一会儿了。当她说话的时候,...
- RabbitMQ 延迟消息实战(rabbitmq如何保证消息不被重复消费)
-
现实生活中有一些场景需要延迟或在特定时间发送消息,例如智能热水器需要30分钟后打开,未支付的订单或发送短信、电子邮件和推送通知下午2:00开始的促销活动。RabbitMQ本身没有直接支持延迟...
- Java对象拷贝原理剖析及最佳实践(java对象拷贝方法)
-
作者:宁海翔1前言对象拷贝,是我们在开发过程中,绕不开的过程,既存在于Po、Dto、Do、Vo各个表现层数据的转换,也存在于系统交互如序列化、反序列化。Java对象拷贝分为深拷贝和浅拷贝,目前常用的...
- 如何将 Qt 3D 渲染与 Qt Quick 2D 元素结合创建太阳系行星元素?
-
Qt组件推荐:QtitanRibbon:遵循MicrosoftRibbonUIParadigmforQt技术的RibbonUI组件,致力于为Windows、Linux和MacOSX提...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)