C标准库<signal.h>实现

简介:

背景知识

signal.h是C标准函数库中的信号处理部分, 定义了程序执行时如何处理不同的信号。信号用作进程间通信, 报告异常行为(如除零)、用户的一些按键组合(如同时按下Ctrl与C键,产生信号SIGINT)。信号是程序执行过程中发生的异常事件,同步信号的产生 是因为程序自身的某些动作,例如除零或不正当的访问存储器,异步信号是由程序外部的行为引起的,比如有人敲击了提示键或者另外一个程序(异步地执行)给你 的程序发信号,都会引发一个异步信号。程序不能屏蔽的信号要求立即得到处理。如果不对发生的信号指定一种处理方法,那么它就会被作为一种致命错误对待,然 后程序就会以失败的状态终止执行

头文件<singal.h>定义了一个无穷信号集的各种编码值:

#define _NSIG             16    //能够使用的信号的数量 

#define SIGHUP             1    // 挂起 
#define SIGINT             2    // 中断 (DEL) 
#define SIGQUIT            3    // 退出 (ASCII FS) 
#define SIGILL             4    // 非法指令 
#define SIGTRAP            5    // 跟踪中断 (捕捉的时候没有重置) 
#define SIGABRT            6    // IOT 指令 
#define SIGIOT             6    // 终止使用PDP-11 
#define SIGUNUSED          7    // 备用代码 
#define SIGFPE             8    // 浮点型异常 
#define SIGKILL            9    // kill (不能捕获或忽略) 
#define SIGUSR1           10    // 用户定义信号 # 1 
#define SIGSEGV           11    // 内存段异常 
#define SIGUSR2           12    // 用户定义信号 # 2 
#define SIGPIPE           13    // 在管道上写操作没有其他的读操作 
#define SIGALRM           14    // 警告 
#define SIGTERM           15    // kill后获得软件终止信号 

#define SIGEMT             7    // 旧式 
#define SIGBUS            10    // 旧式 

/* POSIX 要求定义下面的信号, 即使它们不被支持. 这里给出这些定义, 但是它们不被支持. */
#define SIGCHLD           17    // 子进程终止或停止 
#define SIGCONT           18    // 如果停止则继续 
#define SIGSTOP           19    // 停止信号 
#define SIGTSTP           20    // 交互式停止信号 
#define SIGTTIN           21    // 后台进程试图读 
#define SIGTTOU           22    // 后台进程试图写 

两个函数:

raise函数,报告一个同步信号

int raise (int _sig);

signal函数,可以指定一种信号的处理方法

__sighandler_t signal (int _sig, __sighandler_t _func) ;

可以通过一下三种方式处理信号:

  • 默认处理是终止程序,就像上面所描述的那样

  • 信号忽略是直接把它丢弃

  • 信号处理是把控制权移给一个指定的函数

C标准的内容

头文件 <signal.h>为处理各种各样的信号声明了一个类型和两个函数,并定义了几个宏

定义的类型是:sig_atomic_t

typedef int sig_atomic_t;

该类型是整数类型,声明为这种类型的对象可以作为一个原子实体被访问,即使有异步中断发生的时候也是如此。

定义的宏有: SIG_DFL、SIG_ERR、SIG_IGN

#define SIG_ERR    ((__sighandler_t) -1)    // 出错返回值 
#define SIG_DFL       ((__sighandler_t)  0)    // 默认信号处理 
#define SIG_IGN       ((__sighandler_t)  1)    // 忽略信号 

它们展开为具有不同值的常量表达式,这些表达式的类型和函数signal的第二个参数及返回值的类型兼容,并且它们的值与所有声明的函数的地址都不相等。

指定信号处理

函数signal:

typedef void (*__sighandler_t) (int);
 __sighandler_t signal(int _sig, __sighandler_t _func);

或声明成:

void (*signal(int sig, void (*func) (int)))(int);

signal函数通过3种方式来保证接收到的信号编号sig被依次处理,如果func的值是SIG_DFL,那么会使用默认的信号处理方式;如果 func的值是SIG_IGN,那么这个信号就会被忽略;否则,func就指向一个函数,当这个信号发生时,就调用这个函数,这样的函数就被称为信号处理 程序。

函数raise:

int raise(int sig);

函数raise把信号sig送给正在执行的程序

返回值:执行成功返回0,否则返回非0

<singal.h>的使用

从本质上说,信号处理是不可移植的。只有当必须指定一个已知的操作系统集的信号处理时,才能使用<signal.h>中声明的函数,不用费尽心思把代码变成通用的,这是不可能的

如果对一个信号的默认处理是可行的,那么就选择这种处理方法,使用自己的信号处理程序会降低可移植性并且可能导致程序对信号的误处理增多,如果必须提供某个信号的处理程序,对其进行分类如下:

不可能返回值的信号处理程序,e.g:SIGFPE和SIGABRT

必须返回值的信号处理程序,e.g:SIGINT

不能返回值的信号处理程序最后会调用abort、exit或者longjmp来结束。当然,SIGABRT的处理程序不能调用abort来结束。处 理程序不应该调用singal使自己重建。如果程序没有终止,就把剩下的工作给其他的函数。如果信号是异步的,要注意程序所有的输入输出,因为这样的操作 也会中断库的执行。

必须返回的信号处理程序用一个return语句结束,如果它想让自己复位,那么应该在入口处立即返回。如果信号是异步的,就在一个 sig_atomic_t类型的volatile数据对象中存储一个非零值。不要执行其他任何对正在执行的程序有副作用的动作,例如输入输出和访问其他的 数据对象

 <singal.h>的实现

下面是一个异步信号处理程序的例子:

#include<signal.h>
static sig_atomic_t intflag = 0;
static void field_int(int sig) {
    signal(SIGINT, &field_int);
    intflag = 1;
    return;
}

这个程序调用signal(SIGINT, &field_int)来创建处理程序。它会不时地执行下面的代码来检查是否发生了异步交互式提示中断:

if(intflag) {
    //执行中断
    intflag = 0;
    ...
}

 raise.c 

#include<stdio.h>
#include<signal.h>
#include<stdlib.h>
_sigfun *_sigtable[_NSIG] = {0};
int raise(int sig)
{
    _sigfun *s;
    if(sig <= 0 || _NSIG <= sig)
        return -1;
    if((s = _sigtable[sig]) != SIG_IGN && s!= SIG_DFL) {
        _sigtable[sig] = SIG_DFL;
        (*s)(sig);
    } else if (s == SIG_DFL) {
        char ac[10], *p;
        switch(sig) {
        case SIGABRT:
            p = "abort";
            break;
        case SIGFPE:
            p = "arithmetic error";
            break;
        case SIGILL:
            p = "invalid executable code";
            break;
        case SIGINT:
            p = "interruption";
            break;
        case SIGSEGV:
            p = "invalid storage access";
            break;
        case SIGTERM:
            p = "termination request";
            break;
        default:
            *(p = &ac[sizeof(ac) - 1]) = '\0';
            do
                *--p = sig % 10 + '0';
            while((sig /= 10) != 0);
            fputs("signal #", stderr);
            break;
    }
        fputs(p, strerr);
        fputs(" -- terminating\n", stderr);
        exit(EXIT_FAILURE);
    }
    return 0;
}

 signal.c

#include<signal.h>
_sigfun *_sigtable[_NSIG];
_sigfun *signal(int sig, _sigfun *fun)
{
    _sigfun *s;
    if(sig <= 0 || _NSIG <= sig || fun == SIG_ERR)
        return SIG_ERR;
    s = _sigtable[sig];
    _sigtable[sig] = fun;
    return s;
}
目录
相关文章
|
4月前
|
安全
C 标准库 - <signal.h> 详解
`&lt;signal.h&gt;` 是 C 标准库中的头文件,提供信号处理功能,用于通知程序特定事件,如非法操作或定时器到期。它定义了多种信号常量(如 `SIGINT`、`SIGTERM`、`SIGKILL`、`SIGSEGV`、`SIGUSR1` 和 `SIGUSR2`),并允许通过 `signal()` 或 `sigaction()` 设置信号处理函数。
|
4月前
|
程序员 C++
C 标准库 - <setjmp.h>详解
`&lt;setjmp.h&gt;` 是 C 标准库中的头文件,用于处理程序的非局部跳转。它提供了 `setjmp` 和 `longjmp` 函数,允许程序保存和恢复执行状态,适用于错误处理和复杂控制流(如协程)。主要概念包括跳转和上下文保存。使用时需注意局部变量作用域、不对称性及避免滥用。
75 11
|
4月前
|
安全
C 标准库 - <errno.h>详解
`&lt;errno.h&gt;` 是 C 标准库中的错误处理头文件,定义了全局变量 `errno` 和一系列错误代码宏,如 `ENOMEM`(内存不足)、`EINVAL`(无效参数)等,用于检测和处理函数调用中的错误。在调用可能引发错误的函数前后应检查和清零 `errno` 以准确识别错误原因。
|
7月前
|
存储 程序员 C语言
C 标准库 - <errno.h>
C 标准库 - <errno.h>
|
7月前
|
存储 C语言
C语言标准库介绍:<time.h>
C语言标准库介绍:<time.h>
|
8月前
|
Java API C++
【C++ 与Qt 线程】C++ std::thread 与Qt qthread多线程混合编程
【C++ 与Qt 线程】C++ std::thread 与Qt qthread多线程混合编程
296 1
|
8月前
|
存储 安全 算法
【C++ 包装器类 std::atomic 】全面入门指南:深入理解并掌握C++ std::atomic 原子操作 的实用技巧与应用
【C++ 包装器类 std::atomic 】全面入门指南:深入理解并掌握C++ std::atomic 原子操作 的实用技巧与应用
677 1
|
8月前
|
存储 Unix Linux
<C语言错误处理> <signal.h>头文件及signal函数等内置函数
<C语言错误处理> <signal.h>头文件及signal函数等内置函数
|
存储 固态存储 Linux
深入学习tombstone和signal
深入学习tombstone和signal
深入学习tombstone和signal
|
C++
boost之signal的使用
boost是C++的一个扩展库,被称为C++准标准库,里面的组件很丰富,并且引用方便,85%的组件只需要引用头文件即可使用。在嵌入式系统也可以很方便的使用,这里介绍一下signal的使用,有点类似Qt里的信号槽。 可以接收静态函数、类成员函数、labmda表达式。 下面这个是使用signal封装的一个事件注册处理模板,使用起来还是很方便的。
142 0