良许Linux教程网 干货合集 你真的懂Linux线程和信号的关系吗?

你真的懂Linux线程和信号的关系吗?

引言:

在讨论信号时,我们通常更多地从进程的角度去理解,但当一个进程包含多个线程时,进程、线程、信号之间的关系会是怎样呢?

1. 线程与信号的关联

线程与信号的关系遵循以下几个标准:

  • 标准1:同一线程组的线程共享信号处理函数。
  • 标准2:线程拥有独立的信号阻塞集。
  • 标准3:存在私有未决信号队列和共享未决信号队列。
  • 标准4:当收到致命信号时,线程组将退出。

2.同一线程组线程之间共享信号处理函数(\标准1\)。

创建进程时,线程task_struct对象sighand成员会指向主线程指向的struct sighand_struct对象,线程组所有线程共享主线程信号表。

原来的进程此时理解为主线程。

image-20240414222804149
image-20240414222804149

**3.线程有独立的阻塞信号集(*标准2*)。

每个线程task_struct都有各自的阻塞/屏蔽信号集(blocked成员)。

通过pthread_sigmask可以设置阻塞/屏蔽信号集,用来屏蔽特定的信号,

进程(主线程)可以由sigprocmask函数或者pthread_sigmask函数来设置阻塞/屏蔽信号集。

sigprocmask函数和pthread_sigmask底层实现方式一样,没有区别。

image-20240414222807767
image-20240414222807767

pthread_sigmask函数原型

int pthread_sigmask(int how, const sigset_t *set, sigset_t *oldset);

功能:pthread_sigmask函数用于设置线程阻塞信号集。

参数:

how:表示信号屏蔽字的修改方式。可以取以下三个值之一:

  • SIG_BLOCK:将set指定的信号添加到进程的当前信号屏蔽字中。
  • SIG_UNBLOCK:将set指定的信号从进程的当前信号屏蔽字中移除。
  • SIG_SETMASK:将当前信号屏蔽字设置为set指定的值。

set:指向需要修改的新信号屏蔽字集合的指针。

oldset:指向保存之前信号屏蔽字的集合的指针。

返回值:

成功:返回0。

失败:返回-1,并设置errno。

===========================================

文章没看懂没关系,每篇文章都有视频讲解:

图解Linux环境编程视频课程(完整版)正式发布了!!!

===========================================


**4.私有未决信号队列和共享未决信号队列(*标准3*)。

每个线程task_struct都有各自的未决信号队列(pending成员)。

每个线程组主线程有一个共享未决信号队列(signal成员),其他线程共享主线程共享未决信号队列。

当主线程收到信号后,如果主线程设置了阻塞/屏蔽信号集,那么该信号会存储在共享未决信号队列,比如通过kill或者sigqueue函数发送给主线程的信号。

其他线程收到信号后,如果线程设置了阻塞/屏蔽信号集,那么该信号会存储在私有未决信号队列,比如通过tkill,tgkill,pthread_kill,以及pthread_sigqueue函数发送给线程的信号。

image-20240414222812068
image-20240414222812068

共享未决信号队列信号如何处理?

  • 主线程优先处理共享未决信号队列信号。
  • 其他线程在主线程不方便处理时,才会处理共享未决信号队列信号。

tkill函数原型

int tkill(int tid, int sig);

int tgkill(int tgid, int tid, int sig);

描述:

tkill函数用于发送信号到指定线程,tid表示线程ID,sig表示信号编号,不安全尽量少用。

tgkill函数用于向指定线程发送信号,tgid表示线程组ID,tid表示线程ID,sig表示信号编号。

返回值:

成功:返回0。

失败:返回-1,并设置errno。


pthread_kill函数原型

int pthread_kill(pthread_t thread, int sig);

功能:pthread_kill函数用于向指定线程发送信号的函数。

参数:

thread:为目标线程的标识。

sig:要发送的信号的编号。

返回值:

成功:返回0。

失败:返回errno。


pthread_sigqueue函数原型

int pthread_sigqueue(pthread_t thread, int sig,

    const union sigval value);

功能:pthread_sigqueue函数是一个用于向指定线程发送信号及数据的函数。

pthread_sigqueue使用方法可以参考sigqueue函数。

参数:

thread:为目标线程的线程标识符。

sig:为要发送的信号编号。

value:为发送给目标线程的附加数据。

返回值:

成功:返回0。

失败:返回errno。


常用产生信号函数对比:

image-20240414222816009
image-20240414222816009

5.收到致命信号,线程组退出(标准4)。

所谓致命信号就是能让线程终止的信号,可查看信号默认处理方式表。

这一条标准的目的是为了实现同一线程组里线程的统一退出管理。

以上就是良许教程网为各位朋友分享的Linu系统相关内容。想要了解更多Linux相关知识记得关注公众号“良许Linux”,或扫描下方二维码进行关注,更多干货等着你 !

137e00002230ad9f26e78-265x300
本文由 良许Linux教程网 发布,可自由转载、引用,但需署名作者且注明文章出处。如转载至微信公众号,请在文末添加作者公众号二维码。
良许

作者: 良许

良许,世界500强企业Linux开发工程师,公众号【良许Linux】的作者,全网拥有超30W粉丝。个人标签:创业者,CSDN学院讲师,副业达人,流量玩家,摄影爱好者。
上一篇
下一篇

发表评论

联系我们

联系我们

公众号:良许Linux

在线咨询: QQ交谈

邮箱: yychuyu@163.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

关注微博
返回顶部