C标准库<signal.h>实现
本文地址:http://www.cnblogs.com/archimedes/p/c-library-signal.html,转载请注明源地址。
背景知识
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 = ;
static void field_int(int sig) {
signal(SIGINT, &field_int);
intflag = ;
return;
}
这个程序调用signal(SIGINT, &field_int)来创建处理程序。它会不时地执行下面的代码来检查是否发生了异步交互式提示中断:
if(intflag) {
//执行中断
intflag = ;
...
}
raise.c
#include<stdio.h>
#include<signal.h>
#include<stdlib.h>
_sigfun *_sigtable[_NSIG] = {};
int raise(int sig)
{
_sigfun *s;
if(sig <= || _NSIG <= sig)
return -;
if((s = _sigtable[sig]) != SIG_IGN && s!= SIG_DFL) {
_sigtable[sig] = SIG_DFL;
(*s)(sig);
} else if (s == SIG_DFL) {
char ac[], *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) - ]) = '\0';
do
*--p = sig % + '';
while((sig /= ) != );
fputs("signal #", stderr);
break;
}
fputs(p, strerr);
fputs(" -- terminating\n", stderr);
exit(EXIT_FAILURE);
}
return ;
}
signal.c
#include<signal.h>
_sigfun *_sigtable[_NSIG];
_sigfun *signal(int sig, _sigfun *fun)
{
_sigfun *s;
if(sig <= || _NSIG <= sig || fun == SIG_ERR)
return SIG_ERR;
s = _sigtable[sig];
_sigtable[sig] = fun;
return s;
}
C标准库<signal.h>实现的更多相关文章
- C 标准库 - ctype.h
C 标准库 - ctype.h This header declares a set of functions to classify and transform individual charact ...
- C标准库<ctype.h>实现
本文地址:http://www.cnblogs.com/archimedes/p/c-library-ctype.html,转载请注明源地址. 1.背景知识 ctype.h是C标准函数库中的头文件,定 ...
- C 标准库 - ctype.h之iscntrl 使用
iscntrl int iscntrl ( int c ); Check if character is a control character 检查给定字符是否为控制字符,即编码 0x00-0x1F ...
- C 标准库 - ctype.h之isalpha使用
isalpha int isalpha ( int c ); Checks whether c is an alphabetic letter. 检查给定字符是否字母字符,即是大写字母( ABCDEF ...
- C 标准库 - ctype.h之isalnum使用
isalnum int isalnum ( int c ); Checks whether c is either a decimal digit or an uppercase or lowerca ...
- C 标准库 - string.h
C 标准库 - string.h This header file defines several functions to manipulate C strings and arrays. stri ...
- C 标准库 - <assert.h>
C 标准库 - <assert.h> 简介 C 标准库的 assert.h头文件提供了一个名为 assert 的宏,它可用于验证程序做出的假设,并在假设为假时输出诊断消息. 已定义的宏 a ...
- C 标准库 - <stdarg.h>
C 标准库 - <stdarg.h> 简介 stdarg.h 头文件定义了一个变量类型 va_list 和三个宏,这三个宏可用于在参数个数未知(即参数个数可变)时获取函数中的参数. 可变参 ...
- C 标准库 - <signal.h>
C 标准库 - <signal.h> 简介 signal.h 头文件定义了一个变量类型 sig_atomic_t.两个函数调用和一些宏来处理程序执行期间报告的不同信号. 库变量 下面是头文 ...
- C 标准库 - <setjmp.h>
C 标准库 - <setjmp.h> 简介 setjmp.h 头文件定义了宏 setjmp().函数 longjmp() 和变量类型 jmp_buf,该变量类型会绕过正常的函数调用和返回规 ...
随机推荐
- 【转载】 IE/Firefox每次刷新时自动检查网页更新,无需手动清空缓存的设置方法
[参考了别人的文章]我们做技术,经常在写页面的时候需要多次刷新测试,可是浏览器都有自己的 缓存机制,一般CSS和图片都会被缓存在本地,这样我们修改的CSS就看不到效果 了,每次都去清空缓存,再刷新看效 ...
- VS快捷键的简单总结
一.一般的快捷键 Shift+Alt+Enter: 切换全屏编辑 Ctrl+B,T / Ctrl+K,K: 切换书签开关Ctrl+B,N / Ctrl+K,N: 移动到下一书签Ctrl+B,P: 移动 ...
- DbUtils使用例子
DbUtils: JDBC Utility Component Examples This page provides examples that show how DbUtils may be us ...
- Hadoop第11周练习—HBase基础知识
1 :数据即日志 内容 2 :HBase合并过程 内容 3 :HBase一致性 内容 书面作业1:数据即日志 内容 我们常说HBase是“数据即日志”的数据库,它是怎样修改和删除数据的?和Oracle ...
- 【leetcode】3 SUM
3 SUM 原题: Given an array S of n integers, are there elements a, b, c in S such that a + b + c = 0? F ...
- 浅谈java类集框架和数据结构(1)
在另外一篇博客我简单介绍了java类集框架相关代码和理论. 这一篇博客我主要分析一下各个类集框架的原理以及源码分析. 一:先谈谈LinkedList 这是LinkedList源码的开头,我们能看到几点 ...
- 希望早几年知道的5个Unix命令
原文: http://spin.atomicobject.com/2013/09/09/5-unix-commands/ 希望早几年知道的5个Unix命令 使用*nix系统已经有一段时间了.但是还是有 ...
- java:[1,1] 需要class, interface或enum
状态: cmd编译.java文件时报异常:java:[1,1] 需要class, interface或enum异常原因: 主要原因是java文件的编码问题. 在中文操作系统中,使用一贯的"j ...
- Asp.net生成静态网页的实现代码
现在做程序都要将动态的页面转换成静态页面,今天教大家在ASP.NET 中实现静态页面的生成方法. using System; using System.Data; using System.Con ...
- js禁止复制页面文字
做项目的时候有客户提出要求,不能用用户浏览他发表的文章时复制他的文章 一种比较简单的方法,禁止用户选中页面的文字和禁止用户右键菜单 document.oncontextmenu = new Funct ...