美文网首页
进程间通信--信号2(signal,sigaction)

进程间通信--信号2(signal,sigaction)

作者: 锈色的栅栏 | 来源:发表于2024-11-09 09:58 被阅读0次

一个进程收到信号:执行默认动作、忽略信号、执行自定义动作。
【注意】:SIGKILLSIGSTOP 不能更改信号的处理方式,因为它们向用户提供了一种使进程终止的可靠方法

1、signal函数

#include<signal.h>
typedef void(*sighandler_t)(int);
sighandler_t signal(int signum,sighandler_t handler);

功能:
        注册信号处理函数(不可用于 SIGKILL、SIGSTOP 信号),即确定收到信号后处理函数的入口地址。
        此函数不会阻塞。
参数:
        signum:信号的编号,这里可以填数字编号,也可以填信号的宏定义,可以通过命令kill - l("l" 为字
        母)进行相应查看。
        handler : 取值有 3 种情况:
            SIG_IGN:忽略该信号SIG_DFL:执行系统默认动作信号处理函数名:自定义信号处理函数,如:func
            回调函数的定义如下:
            void func(int signo)
            {
                // signo 为触发的信号,为 signal() 第一个参数的值
            }
返回值:
        成功:第一次返回NULL,下一次返回此信号上一次注册的信号处理函数的地址。如果需要使
        用此返回值,必须在前面先声明此函数指针的类型。
        失败:返回 SIG_ERR

2、sigaction函数

#include<signal.h>
int sigaction(int signum,const struct sigaction *act,struct sigaction*oldact);

功能:
    检查或修改指定信号的设置(或同时执行这两种操作)。
参数:
    signum:要操作的信号。
    act: 要设置的对信号的新处理方式(传入参数)。
    oldact:原来对信号的处理方式(传出参数)。
    如果 act 指针非空,则要改变指定信号的处理方式(设置),如果 oldact 指针非空,则系统将此前指定
    信号的处理方式存入 oldact。
返回值:
    成功:0
    失败:-1
struct sigaction结构体:

struct sigaction{
    void(*sa_handler)(int); //旧的信号处理函数指针
    void(*sa_sigaction)(int, siginfo_t *, void *); //新的信号处理函数指针
    sigset_t sa_mask; //信号阻塞集
    int sa_flags; //信号处理的方式
    void(*sa_restorer)(void); //已弃用
};

1) sahandler、sasigaction:信号处理函数指针,和 signal() 里的函数指针用法一样,应根据情况给 sasigaction、sahandler 两者之一赋值,其取值如下:
    a)SIGIGN:忽略该信号 b) SIGDFL:执行系统默认动作 c) 处理函数名:自定义信号处理函数
2) samask:信号阻塞集,在信号处理函数执行过程中,临时屏蔽指定的信号。
3) saflags:用于指定信号处理的行为,通常设置为 0,表使用默认属性。它可以是一下值的“按位或”组合:
        Ø SARESTART:使被信号打断的系统调用自动重新发起(已经废弃)
        Ø SANOCLDSTOP:使父进程在它的子进程暂停或继续运行时不会收到 SIGCHLD 信号。
        Ø SANOCLDWAIT:使父进程在它的子进程退出时不会收到 SIGCHLD 信号,这时子进程如果退
        出也不会成为僵尸进程。
        ØSANODEFER:使对信号的屏蔽无效,即在信号处理函数执行期间仍能发出这个信号。
        Ø SARESETHAND:信号处理之后重新设置为默认的处理方式。
        ØSASIGINFO:使用 sasigaction 成员而不是 sahandler 作为信号处理函数。

信号处理函数:
    void(*sa_sigaction)(int signum, siginfo_t *info, void *context);
参数说明:
    signum:信号的编号。
    info:记录信号发送进程信息的结构体。
    context:可以赋给指向 ucontext_t 类型的一个对象的指针,以引用在传递信号
                 时被中断的接收进程或线程的上下文。

示例代码

#include<stdio.h>
#include<signal.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/time.h>

void my_func(int sig)
{
    printf("ctrl+c被按下了\n");
}
int main()
{
    struct sigaction act;
    //act存放回调函数
    act.sa_handler = my_func;
    //act添加阻塞集 act.sa_mask
    //清空阻塞集
    //sigemptyset(&act.sa_mask);
    //将所有信号添加到阻塞集中
    sigfillset(&act.sa_mask);
    //SA_RESETHAND:信号处理之后重新设置为默认的处理方式
    act.sa_flags = 0; //默认方式
    //act.sa_flags |= SA_RESETHAND;

    sigaction(SIGINT, &act, NULL);
    while (1);
    return 0;
}

相关文章

网友评论

      本文标题:进程间通信--信号2(signal,sigaction)

      本文链接:https://www.haomeiwen.com/subject/svkadjtx.html