当前位置: 首页 > news >正文

Linux系统编程 day7、8 信号(周日划水了)

信号相关概念

信号这章难就难在其抽象。

信号共性:简单、不能携带大量数据、满足条件才发送。

信号的特质:信号是软件层面上的“中断”,一旦信号产生,无论程序执行到什么位置,必须立即停止,处理信号,处理结束再继续执行后续指令。所有信号产生及处理都是由内核完成的。

产生信号方式:1、按键产生 2、系统调用产生 3、软件条件产生 4、硬件异常产生 5、命令产生

未决:产生与递达之间的状态

递达:产生并且送达到进程。直接被内核处理。

信号处理方式:执行默认处理动作、忽略、捕捉(自定义)

阻塞信号集(信号屏蔽字):本质:位图,用来记录信号的屏蔽状态,在解除屏蔽之前,一直处于未决态。

未决信号集:本质:位图。用来记录信号的处理状态。该信号集中的信号表示已经产生,但尚未递达。

查看系统信号表kill -l      1-31常规信号,之后的是实时信号一般在底层驱动的时候会用

信号四要素

信号使用之前,应先确定四要素      1、编号    2、名称    3、事件    4、默认处理动作   

kill命令和kill函数  

发送SIGKILL信号

#include<signal.h>
int kill(pid_t pid , signal);
signal:要直接用宏的名字不应该用编号 , eg. SIGKILL
pid > 0 指定进程
pid = 0 指定同一进程组的所有进程
pid = -1 发送给进程有权限发送的系统中的所有进程
pid < -1 取|pid|发给对应进程组成功 0 
失败 -1

alarm函数:使用自然计时法

设定定时器,指定seconds之后,内核会给当前进程发送SIGALRM信号,进程收到该信号,默认动作终止。

每个进程都有且只有唯一一个定时器。

取消定时器alarm(0) , 返回旧定时器剩余秒数。

unsigned int alarm(unsigned int seconds);
seconds:定时秒数
返回值:上次定时剩余时间
无错误现象

可以使用time命令查看程序运行时间,实际时间 = 用户时间+内核时间+等待时间

--->程序运行的瓶颈在于IO,优化程序,首选优化IO。

setitimer函数

可以实现周期定时。

#include<sys/time.h>
int setitimer(int which , const struct itimerval *new_value , struct itimerval *old_value);
参数:which指定定时方式
1、自然定时:ITIMER_REAL ->14.SIGLARM    计算自然时间   √
2、虚拟空间计时(用户空间计时):ITIMER_VIRTUAL ->26. SIGVTALRM    计算进程占用cpu时间
3、运行时计时(用户+内核):ITIMER_PROF ->27.SIGPROF    计算占用cpu及执行系统调用时间
返回值:成功 0失败 -1new_value : 定时秒数类型:struct itimerval {struct timeval struct timeval {time_t      tv_sec;         /* seconds */suseconds_t tv_usec;        /* microseconds */};it_interval; /* Interval for periodic timer */  ---->设定两次定时任务之间的时间间隔,即周期定时秒数struct timeval struct timeval {time_t      tv_sec;         /* seconds */suseconds_t tv_usec;        /* microseconds */};it_value;    /* Time until next expiration */}; --->第一次定时的时长
初始化:struct itimerval new_time;
new_time.it_interval.tv_sec = 1;
new_time.it_interval.tv_usec = 0;
new_time.it_value.tv_sec = 0;
new_time.it_value.tv_usec = 0;
old_value : 传出参数,上次定时剩余时间

 信号集操作函数(重点)

阻塞信号集可以进行操作用来影响未决信号集。未决信号集不能直接操作。

sigset_t set;// 自定义信号集
sigemptyset(sigset_t *set);// 清空信号集
sigfillset(sigset_t *set);// 全部置1
sigaddset(sigset_t *set , int signum);// 将一个信号添加到集合中
sigdelset(sigset_t *set , int signum);// 将一个信号从集合中移除
成功返回0 失败返回-1sigismember(const sigset_t *set , int signum);// 判断一个信号是否在集合中
在--> 返回1  不在-->返回0  

 sigprocmask函数

用来屏蔽信号、解除信号。其本质,读取或修改进程的信号屏蔽字。 

int sigprocmask(int how, const old_kernel_sigset_t *set, old_kernel_sigset_t *oldset);how取值:
SIG_BLOCK:设置阻塞
SIG_UNBLOCK:取消阻塞
SIG_SETMASK:用自定义的set替换maskset:自定义set
oldset:旧有的mask成功0 失败-1

sigpending函数

 查看未决信号集

int sigpending(sigset_t *set);
返回值:成功返回0,失败返回-1
sigset_t *set 传出参数   传出的未决信号集

总结 

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
#include<signal.h>void printset(sigset_t* set)
{int i;for(i = 0 ; i < 32 ; i++){if(sigismember(set , i)){putchar('1');}else{putchar('0');}}printf("\n");
}int main(int argc , char *argv[])
{sigset_t set , oldset , newset;sigemptyset(&set) ; //初始化setsigaddset(&set , SIGINT) ; // 将SIGINT位置变为1int ret = sigprocmask(SIG_BLOCK , &set , &oldset);if(ret == -1){perror("sig error");exit(1);}while(1){sigpending(&newset);printset(&newset);sleep(1);}return 0;    
}

信号捕捉

signal函数

注册一个信号捕捉函数。

signal函数第二个参数是函数,必须是void类型,且输入参数为int的函数

typedef void(*sighandler_t)(int);sighandler_t signal(int signum , sighandler_t handler);signum :信号 SIGINT
后面是函数,必须是void类型,且输入参数为int的函数
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
#include<signal.h>void catch(int signo)
{printf("catch you!!!%d\n" , signo);
}int main(int argc , char *argv[])
{signal(SIGINT , catch);while(1);return 0;
}

sigaction函数(重点)

注册一个信号捕捉函数。

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);struct sigaction {void     (*sa_handler)(int);  //捕捉到了去执行一个东西void     (*sa_sigaction)(int, siginfo_t *, void *); //不用sigset_t   sa_mask; //重点,这个mask只作用于捕捉函数调用期间 绝大多数传0int        sa_flags;  //设置的参数  绝大多数传0void     (*sa_restorer)(void); //废弃};返回值:成功0 失败-1
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
#include<signal.h>void catch(int signo)
{printf("catch you!!!%d\n" , signo);
}int main(int argc , char *argv[])
{struct sigaction act , oldact;act.sa_handler(catch);    // 回调函数,信号捕捉函数sigemptyset(act.sa_mask);     //清空sa_mask   act.sa_flags = 0;    //设置属性 , 一般用0;int ret = sigaction(SIGINT , &act , &oldact);if(ret == -1){perror("sig error");exit(1);}while(1);return 0;
}

捕捉函数特性

1、捕捉函数执行期间,信号屏蔽字由mask变成sa_mask,执行结束恢复为mask。

2、信号捕捉函数执行期间,本信号自动被屏蔽(sa_flags = 0)。

3、阻塞的常规信号不支持排队,产生多次只记录一次(后32个实时信号支持排队)。

相关文章:

  • LangChain + 文档处理:构建智能文档问答系统 RAG 的实战指南
  • Hyperlane:Rust Web框架的性能新标杆
  • 超详细mac上用nvm安装node环境,配置npm
  • 智慧能源安全新纪元:当能源监测遇上视频联网的无限可能
  • kafka监控kafka manager(CMAK)部署配置
  • Moritex大靶面远心镜头 助力晶圆外观检测
  • 榕壹云预约咨询系统:基于ThinkPHP+MySQL+UniApp打造的灵活预约小程序解决方案
  • 【日志体系】ELK Stack与云原生日志服务
  • 【计算机视觉】CV实战项目- CMU目标检测与跟踪系统 Object Detection Tracking for Surveillance Video
  • 24. git revert
  • Spring(第一章)
  • 11-DevOps-Jenkins Pipeline流水线作业
  • 剑指offer经典题目(五)
  • ORION:通过视觉-语言指令动作生成的一个整体端到端自动驾驶框架
  • WPF的发展历程
  • 人类行为的原动力是自我保存-来自ChatGPT
  • 分布式数据库TiDB:架构、核心特性与生产实践(分库分表)
  • 纷析云开源财务软件:助力企业实现数字化自主权
  • 宝塔面板引发的血案:onlyoffice协作空间无法正常安装的案例分享
  • 树莓派5+L298N控制电机
  • 宇树的任务已经完成?王兴兴也在等待行业拐点
  • 安且吉兮,西泠印社雅集吴昌硕故里
  • 电子产品已拆封,还能申请“七天无理由退货”吗?上海法院这样判
  • 复旦大学史地学系在北碚
  • 我国自主研制的重大航空装备AG600性能怎样?专家解读
  • 3月赴美外国游客数量加速下滑