Linux/UNIX之信号(2)
信号(2)
sigaction函数
sigaction函数的功能是检查或改动与制定信号相关联的处理动作。此函数代替了signal函数。
#include <signal.h>
int sigaction(int signum, const structsigaction *act, struct sigaction *oldact);
此函数使用下列结构:
struct sigaction {
void (*sa_handler)(int);
void (*sa_sigaction)(int, siginfo_t *, void *);
sigset_t sa_mask;
int sa_flags;
void (*sa_restorer)(void);
};
返回值:若成功则返回0,如出错则返回-1。
当中參数,signum是要检測或改动其详细动作的信号编号。若oact指针非空,则将把原先对该信号的动作写到它指向的位置。若act是空指针,则sigaction函数就不须要再做其它设置了,否则将在该參数中设置对指定信号的动作。
在參数act指向的sigaction结构中,sa_handler是一个函数指针,它指向接收信号sig时将被调用的信号处理函数。它相当于前面见到的传递函数signal的參数func。我们能够将sa_handler字段设置为特殊SIG_IGN和SIG_DFL,它们分别表示信号将被忽略或把对该信号的处理方式恢复为默认动作。
sa_mask字段指定了一个信号集,在调用sa_handle所指向的信号处理函数之前,该信号集将被增加到进程的信号屏蔽字中。这是一组将被堵塞且不会被传递给该进程的信号。设置信号屏蔽字能够防止前面看到的信号在它的处理函数还未执行结束时就被接收到的情况。
以下的程序能够实现signal函数。
Sigfunc *
signal(int signo, Sigfunc *func)
{
struct sigaction act, oact; act.sa_handler = func;
sigemptyset(&act.sa_mask);
act.sa_flags = 0;
if (signo == SIGALRM) {
#ifdef SA_INTERRUPT
act.sa_flags |= SA_INTERRUPT;
#endif
}else {
#ifdef SA_RESTART
act.sa_flags |= SA_RESTART;
#endif
}
if (sigaction(signo, &act, &oact) < 0)
return(SIG_ERR);
return(oact.sa_handler);
}
sigsetjmp和siglongjmp
在信号处理中进行非局部转移时,应当使用这两个函数。
#include <setjmp.h>
int sigsetjmp(sigjmp_buf env, intsavesigs);
void siglongjmp(sigjmp_buf env, int val);
这两个函数与setjmp和longjmp之间的唯一差别是sigsetjmp添加了一个參数。假设savesigs非0,则sigsetjmp在env中保存了进程的当前信号屏蔽字。调用siglongjmp时,假设带非0 savesigs的sigsetjmp调用已经保存了env,则siglongjmp从当中恢复保存的信号屏蔽字。
sigsuspend函数
#include<signal.h>
intsigsuspend(const sigset_t *mask);
将进程的信号屏蔽字设置为mask指向的值。在捕捉到一个信号或发生了一个会终止该进程的信号之前,该进程被挂起。假设捕捉到一个信号并且从该信号处理程序返回,则sigsuspend返回,并且将该进程的信号屏蔽字设置为调用sigsuspend之前的值。
absorb函数
#include<stdlib.h>
voidabort(void);
该函数使异常终止。此函数将SIGABRT信号发送给调用进程。调用abort将向主机环境传递一个为成功终止的通知,其方法是调用raise函数。abort将会冲洗所以打开的流;确保进程对此信号不堵塞,不理会进程对此信号的堵塞。
下面是POSIX.1中abort函数的实现:
#include<signal.h>
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h> void
abort(void) /* POSIX-style abort() function */
{
sigset_t mask;
struct sigaction action; /*
* Caller can't ignore SIGABRT, if so resetto default.
*/
sigaction(SIGABRT, NULL, &action);
if (action.sa_handler == SIG_IGN) {
action.sa_handler = SIG_DFL;
sigaction(SIGABRT, &action, NULL);
}
if (action.sa_handler == SIG_DFL)
fflush(NULL); /* flush all open stdio streams */
/*
* Caller can't block SIGABRT; make sureit's unblocked.
*/
sigfillset(&mask);
sigdelset(&mask, SIGABRT); /* mask has only SIGABRT turned off */
sigprocmask(SIG_SETMASK, &mask, NULL);
kill(getpid(), SIGABRT); /* send the signal */ /*
* If we're here, process caught SIGABRTand returned.
*/
fflush(NULL); /* flush all open stdio streams*/
action.sa_handler = SIG_DFL;
sigaction(SIGABRT, &action, NULL); /* reset to default */
sigprocmask(SIG_SETMASK, &mask,NULL); /* just in case ... */
kill(getpid(), SIGABRT); /* and one more time */
exit(1); /* this should never be executed ... */
}
sleep函数
#include <unistd.h>
unsigned int sleep(unsigned int seconds);
此函数使调用进程被挂起,直到满足下面条件之中的一个:
1. 已经过了seconds所指定的墙上时钟时间
2. 调用进程捕捉到一个信号并从信号处理程序返回
作业控制信号
6个与作业控制有关的信号:
SIGCHLD 子进程已停止或终止
SIGCONT 假设进程已停止,则使其继续执行
SIGSTOP 停止信号
SIGTSTP 交互式停止信号
SIGTTIN 后台进程组成员读控制终端
SIGTTOU 后台进程组写到控制终端
当键入挂起字符键(一般是ctrl+Z)时,SIGTSTP被送至前台进程组的全部进程。当我们通知shell在前台或后台恢复执行一个作业时,shell向该作业中的全部进程发送SIGCONT信号。
Linux/UNIX之信号(2)的更多相关文章
- Linux/UNIX之信号(1)
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/walkerkalr/article/details/24462723 信号(1) 信号是软件中断.每 ...
- Linux/Unix监控其他用户和信号
--Linux/Unix监控其他用户和信号 ------------------------------------------------------2013/10/27 查看有哪些用户登录 w ...
- 《Linux/UNIX系统编程手册》第63章 IO多路复用、信号驱动IO以及epoll
关键词:fasync_helper.kill_async.sigsuspend.sigaction.fcntl.F_SETOWN_EX.F_SETSIG.select().poll().poll_wa ...
- [转]Linux进程间通信——使用信号
转载于:http://blog.csdn.net/ljianhui/article/details/10128731 经典!!! Linux进程间通信——使用信号 一.什么是信号 用过 ...
- 《Linux/Unix系统编程手册》读书笔记7 (/proc文件的简介和运用)
<Linux/Unix系统编程手册>读书笔记 目录 第11章 这章主要讲了关于Linux和UNIX的系统资源的限制. 关于限制都存在一个最小值,这些最小值为<limits.h> ...
- 练习--LINUX进程间通信之信号SIGNAL
同样的,信号也不要太迷信可靠信号及不及靠信号,实时或非实时信号. 但必须要了解这些信号之间的差异,函数升级及参数,才能熟练运用. ~~~~~~~~~~~~~~~~ 信号本质 信号是在软件层次上对中断机 ...
- 编写Linux/Unix守护进程
原文: http://www.cnblogs.com/haimingwey/archive/2012/04/25/2470190.html 守护进程在Linux/Unix系统中有着广泛的应用.有时,开 ...
- 转:linux/unix命令行终端的光标及字符控制快捷键
from:http://linux.chinaunix.net/techdoc/system/2007/11/23/973027.shtml 在使用linux/unix的命令行终端时,有时候会碰到键盘 ...
- 《Linux/Unix系统编程手册》 时间子系统
Linux下操作系统编程有两本经典APUE即<Advanced Programming in the UNIX Environment>和TLPI<The Linux Program ...
随机推荐
- 页面提交进不了Action的原因
1.进不了action,页面没有任何js报错,可能的原因是数据类型不一致.例如用ajax方式提交所带的参数类型a是String类型,而action中定义的a是Integer类型就会导致这种情况的发生.
- [置顶] Guava学习之Lists
Lists类主要提供了对List类的子类构造以及操作的静态方法.在Lists类中支持构造ArrayList.LinkedList以及newCopyOnWriteArrayList对象的方法.其中提供了 ...
- ZOJ 3790 Consecutive Blocks 模拟题
problemCode=3790">Consecutive Blocks 先离散一下,然后模拟,把一种颜色i所在的位置都放入G[i]中.然后枚举一下终点位置,滑动窗体使得起点和终点间花 ...
- HDU 2045 不easy系列之(3)—— LELE的RPG难题
思路: 1.若前n-1位涂的颜色是符合条件的,则因为首尾不同,再加入一位时,仅仅有1种方法:即s[n] = s[n-1] 2.若前n-1位组成的串不符合,再加入一位后合法.即由于首尾同样而引起的不合法 ...
- 在纯C工程的main函数之前跑代码(手工找到程序入口点, 替换为我们自己的函数)
在main函数之前跑代码的方法 方法: 手工找到程序入口点, 替换为我们自己的函数 写测试程序 // test.cpp : Defines the entry point for the consol ...
- 极度简约 最小 Linux 发行版 Tiny Core Linux 7.1 发布
感谢Linux Story的投递 Tiny Core Linux 是一个极度简约但是也高度可扩展的 GNU/Linux 发行版,其之精简甚至可以小到只有 10MB 大小,昨天 5月23日刚刚发布的 T ...
- Not able to reset SmartRF04DD
今天在使用使用CC2540的时候,想下载个程序到CC2540底板上,结果出现Not able to reset SmartRF04DD的错误.如下图 经过一番摸索,最终是按下CCDEBUG上的rese ...
- SRM 582 Div II Level Two SpaceWarDiv2
题目来源:http://community.topcoder.com/stat?c=problem_statement&pm=12556 #include <iostream> # ...
- 有关java中的final关键字
在java中,可能使用到final关键字修饰的有数据.方法和类. 一.final 修饰数据 有final修饰的数据是用来告诉编译器一块数据是恒定不变的,有时数据恒定不变是很有用的,比如: 1.一个永不 ...
- Android broadcast
发送广播而且接受.发送两个广播 Intent intent = new Intent(); intent.setAction("com.wxq.CUSTOM_INTENT"); s ...