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

Linux高阶——1103——Signal信号机制

1、信号机制

在linux和unix系统下,如果想要处置(挂起,结束)进程,可以使用信号,经典消息机制,所以进程包括系统进程都是利用信号处置进程的

kill -l——查看所有系统支持的信号

1-31号信号——Unix经典信号,是软件研发工程师需要了解的信号

32-33号信号——被系统隐藏,归NPTL线程库使用,不开放

34-64号信号——自定义信号,驱动研发了解

2、触发信号的5种方式

ps aux——查看进程信息

1、终端组合按键触发信号

ctl+c——发2号信号SIGINT

ctl+\——发3号信号SIGQUIT

ctl+z——发20号信号SIGTSTP

唤醒——fg 进程号

2、命令触发信号

kill -signo pid——可以向任意进程发出任意信号

3、函数触发信号

#include<signal.h>

kill(pid_t pid,int signo);——向任意进程发送任意信号

raise(int signo);——向调用进程发送任意信号

abort(void)——向调用进程发送SIG_ABRT信号

./mykill 9 1000

mykill代码

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


int main(int argc,char**argv)
{
    if(argc<3)
    {   
        printf("mykill failed:pram error\n");
        exit(0);
    }   
    kill(atoi(argv[2]),atoi(argv[1]));
    return 0;
}

4、硬件异常系统触发信号

当进程执行时,如果违规访问硬件,系统会发送对应的信号,杀死进程,避免访问

1、当进程中出现进程访问权限异常,例如只读内存写操作,产生段错误(内存核心已转储),系统发送SIGSEGV(11号)信号杀死目标进程

2、进程访问内存越界,触发总线错误(内存核心已转储),系统发送SIGBUS(7号)信号,杀死目标进程

3、进程使用cpu,计数异常,系统抛出浮点数例外(核心已转储),系统发送SIGFPE(8号)信号,杀死目标进程

5、软条件触发信号

在使用某个功能或设备时,当使用满足了软条件,系统会发送信号杀死进程

eg:

1、alarm定时器

用户可以自定义定时的时长,当定时的时间达到,系统会发送SIGALRM(14号)信号,但是信号会杀死进程

alarm定时器

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

int main()
{
    int count=0;
    alarm(1);
    while(1)
    {   
        printf("++count %d\n",++count);
    }   
    return 0;
}

2、管道

管道读端关闭,写端向管道写数据,触发软条件,系统向写端发送SIGPIPE(13号)杀死进程

3、信号的行为或动作

系统中提供的信号一般都是用来杀死或挂起进程,信号的行为一般是被提前预设好的,但是可以修改,用户可以自定义变更行为和动作

SIGNAL一般有三种行为

1、SIG_DFL(默认行为:杀死目标)

是信号首选的默认行为

默认处理动作一般分为5种:

TERM——杀死进程

CORE——杀死进程并转储核心

IGN——不干预进程

STOP——挂起进程

COUNT——唤醒进程

2、SIG_IGN(忽略行为:放弃)

处理动作:NULL

3、SIG_ACTION(捕捉行为:自定义)

处理动作:自定义动作

捕捉函数

void sig_do(int n)
{
    printf("gfytguhbjgv"\n);
}

捕捉函数需要开发者自行定义与实现,但是只有信号触发抵达进程,系统自动调用

行为1中的忽略动作与行为2的忽略行为相比,结果一致,但是优先级不同,行为的优先级比动作高

每个信号拥有自己的信号行为结构体,struct sigaction,自己定义修改结构体,对进程原有的结构体进行替换,因为所有默认信号行为结构体中的行为都是默认的

处理行为设定

act.sa_handler=SIG_DFL/SIG_IGN/捕捉函数地址

默认选项

act.sa_flags=0;

临时屏蔽字,可以使用sigemptyset初始化

act.sa_mask;

结构体

struct sigaction act,oact;
void (*sa_handler)(int)
{
    act.sa_handler=SIG_DFL/SIG_IGN/捕捉函数地址
    act.sa_flags=0;
    act.sa_mask;
}

4、改变信号行为的方法

用自定义结构体替换原有的结构体

第一个参数是要修改的信号

第二个参数是新的结构体

第三个参数是原有的结构体

sigaction(SIGINT,&act,&oact);

替换修改信号行为结构体,一般情况下要传出进程原有的结构体,便于复位

虽然SIGINT代表的是2号信号,但是不能只传入2,因为不同电脑的信号名和编号不一定相同

eg:改变信号行为

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

void sig_do(int n)
{
    int flags=3;
    while(flags--)
    {   
        printf("         flag %d running...\n",flags);
        sleep(1);
    }   
}

int main()
{
    struct sigaction act,oact;
    act.sa_handler=sig_do;
    act.sa_flags=0;
    sigemptyset(&act.sa_mask);

    sigaction(SIGINT,&act,&oact);
    while(1)
        sleep(1);
}

改变后不能使用ctrl+c杀死程序

5、信号的传递与处理流程

1、信号由系统发出,传递到PCB

2、PCB中有一个位图,为64位,每一位对应一个信号,称为未决信号集

还有一个位图,为64位,每一位对应一个信号,称为屏蔽字

3、信号从系统发出后,还未到达未决信号集前称为待传递信号

信号能否通过未决信号集取决于未决信号集中的位码,1为不允许,0为允许

如果信号不被允许通过未决信号集,则该信号会被直接丢弃,放弃处理

4、如果信号通过了未决信号集,但还未抵达屏蔽字集,称为未决态信号,都是待处理信号

如果信号通过了未决信号集,系统会将对应的未决信号集的位码置1,如果有相同信号再次被传递过来,会被直接丢弃,因为系统拒绝相同的信号同时传递,未决信号集的变化是系统进程设置的

5、如果信号通过了屏蔽字集,称为递达态信号

当信号变为递达态后,未决信号集的位码会被归0

系统初始是避免相同信号递达的,因为会产生捕捉函数调用冲突,因此如果某个信号递达,系统会设置临时屏蔽,将对应的屏蔽字位置设置为1,当递达信号处理完毕后,再解除屏蔽

当信号变为递达态信号,则信号正在处理,进程将被处置,杀死或者挂起(见上面3)

6、信号阻塞

指的是信号通过了未决信号集,但是未通过屏蔽字

屏蔽字是开发者可以自行设置的,用于屏蔽阻塞某些特定信号

7、如果同时发多个信号,则系统只能处理前两个发过来的信号,其余的会被全部丢弃

因为屏蔽字会被临时置1,递达态信号处理完毕后,才会归0

8、Unix经典信号,不支持排队,但是可以多阻塞1个

9、自定义信号(34-64号)支持排队

因为经典信号(1-31号)的设计初衷是处置进程

自定义信号(34-64号)被设计出来为了软硬件设计,如:磁盘探针寻址寻道,用在上一页或下一页等操作上


http://www.kler.cn/a/381053.html

相关文章:

  • 1 软件工程——概述
  • MFC/C++学习系列之简单记录9——简单加法
  • Go web 开发框架 Iris
  • echarts画风向杆
  • 砂轮磨料基础知识及发展学习笔记
  • 人工智能ACA(四)--机器学习基础
  • 【Stable Diffusion】
  • 家具组装行业产品说明书的创新与优化
  • 鸿蒙笔记--tsets
  • 探索 Move 编程语言:智能合约开发的新纪元
  • CSRF初级靶场
  • 文件操作:使用ByteArrayInputStream
  • A010-基于SpringBoot的宠物健康咨询系统的设计与实现
  • 【LeetCode】【算法】739. 每日温度
  • Harmony项目基础
  • 基于 RNN 的语言模型
  • windows 文件监控 c++ 11及以上版本可用
  • 接口测试(十一)jmeter——断言
  • 力扣最热一百题——验证二叉搜索树
  • 计算机存储单元bit。不同编程语言类型差异。
  • Python面向对象:类和对象的基本操作
  • 在gitlab,把新分支替换成master分支
  • LeetCode 3165.不包含相邻元素的子序列的最大和:单点修改的线段树(动态规划)
  • easyui +vue v-slot 注意事项
  • Grafana+Prometheus监控篇-Nginx
  • C#中,Thread和Task的区别