核心转储的作用
方便异常后,进行调试
为了让代码从release变为debug,所以在makefile中 加入 -g
如果不懂请看 : gdb调试器的使用
输入 gdb 可执行程序 进入gdb调试器
再次输入 core-file +core文件
gdb直接定位到当前进程终止是因为8号信号,信号的更详细描述为 Arithmetic exception
core文件的作用:
不用自己定位了,有gdb自动定位,事后调试
核心转储为什么一般都是被关闭的?
云服务器属于生产环境即测试测过以后真正的做服务的
core.6288文件的大小为232字节,核心转储的文件往往比较大一些
线上部署的某种服务可能会挂掉,不断进行挂掉重启就会不断形成core dump文件,就有可能导致主机挂掉
2.信号保存
1. 概念
1.实际执行信号的处理动作被称为 信号递达
2.信号从产生到递达之间的状态,称为信号未决
3.进程可以选择 阻塞某个信号
假设你不太喜欢一个老师,所以当一个老师留作业时,你只是把作业是什么记录下来,因为你当前正在上课,没有时间去写作业,只有当下午找个时间去写作业
老师布置作业的行为就是操作系统发信号的过程,你作为一个进程,当前因为做优先级更高的事情正在上课,所以没有时间处理信号,只能把作业记下来,等有时间在写作业 ,即递达信号
4.被阻塞的信号产生时将保持未决状态,直到进程解除对此信号的阻塞才执行的递达动作
假设有老师ABC,每个人都留了作业,由于老师AB对你很好,所以你打算先写AB老师留的作业,但是你不太喜欢老师C,所以就不愿意老师C的作业,宁愿去打游戏,所以你把老师C给你留的作业未决了
由于你并不会递达它,所以你把老师C的作业阻塞了
突然有一天,你喜欢老师C了,所以开始想写老师C留的作业,即解除阻塞,写完作业即递达
5.阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后的可选的一种处理工作
同样有一个老师,布置作业后,你记录下来了,可是这个老师平时不查作业,所以直接把这个作业划掉,默认写完了,即忽略该信号
忽略是把作业划掉,默认写完了也就完成了递达动作
而阻塞是把作业记录下来了,不想去写作业,即没有完成也就没有递达动作
2. 信号列表
pending 表:位图结构
比特位的位置表示哪一个信号
比特位的内容表示是否收到该信号
如:00000000…0001000 代表收到4号信号
bolck 表:位图结构
比特位的位置表示哪一个信号
比特位的内容代表是否对应的信号被阻塞
如:0000000…0010 代表2号信号被屏蔽
handler表:函数指针数组
返回值为void,参数为int的函数指针
该数组的下标表示信号编号
数组的特定下标的内容表示该信号的递达动作
3. 信号处理动作
除了自定义捕捉外,还有SIG_DFL(默认动作)与SIG_IGN(忽略信号)
把0强制转化成函数指针类型 即默认情况 终止进程
对2号信号进行SIG_DFL即默认处理
运行可执行程序后,使用2号信号可终止进程
把1强制转化成函数指针类型 即忽略信号
对2号信号做忽略
忽略信号,所以对其做什么动作都没有用了
4.sigset_t
siget_t 用来控制block和pending两张位图表的
控制block表称之为信号屏蔽字,控制pending表称之为pending信号集
sigset_t 是一种位图结构,由操作系统提供的
5. 信号集操作函数
对信号集进行操作
#include <signal.h> int sigemptyset(sigset_t *set); int sigfillset(sigset_t *set); int sigaddset (sigset_t *set, int signo); int sigdelset(sigset_t *set, int signo); int sigismember(const sigset_t *set, int signo);
函数sigemptyset初始化set所指向的信号集,
使其中所有信号的对应bit清零,表示该信号集不包含 任何有效信号
函数sigfillset初始化set所指向的信号集,
使其中所有信号的对应bit置位,表示 该信号集的有效信号包括系统支持的所有信号
注意,在使用sigset_ t类型的变量之前,一定要调 用sigemptyset或sigfillset做初始化,使信号集处于确定的状态。初始化sigset_t变量之后就可以在调用sigaddset和sigdelset在该信号集中添加或删除某种有效信号
sigprocmask
读取/更改进程的信号屏蔽字,即可以更改block这张位图
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
若操作成功返回0 ,否则返回-1
how和set都是输入型参数,oset为输出型参数
oset:当重新设置信号屏蔽字时,一定是对老的block做各种修改,修改之前把老的block通过oset返回,然后才能设置
how表示 想怎么改
共有三个选项
SIG_BLOCK :把第二个参数 set 所指定的信号全部添加到内核的block表中
SIG_UNBLOCK:从内核block表(用于信号屏蔽) 中把指定的若干个信号去掉
SIG_SETMASK:设置当前信号屏蔽字为set指定的值,相当于传什么就设置什么
bolck位图是为了判断信号是否被屏蔽
老的信号屏蔽字为默认动作终止进程,所以block位图全是零
输入ctrl c没有反应,因为使用sigprocmask将set集合中的信号屏蔽了,
而set信号集中就包括2号信号
sigpending
输入 man sigpending
用该系统调用,获取调用进程的pending位图
调用成功返回0,出错返回-1
pending位图表示是否收到信号
运行可执行程序后,刚开始因为没有信号,所以pending表都是0,
在使用2号信号想要干掉进程时,由于2号信号被阻塞, 无法终止进程 并且pending表中对应的2号信号的比特位出现1
若解除对于2号信号的屏蔽,则输入2号信号,会立即进入递达动作
刚开始执行可执行程序时,由于没有信号输入,所以pending表全部为0,
先打印,直到循环10次才解除对信号的屏蔽,最后显示打印的这句话后,执行2号进程默认动作即终止进程