一 信号的基本概念

  信号:是向进程发送的软件通知,通知进程有事件发生。

  生成:表示一个信号的产生。

  捕获:表示接收到一个信号。

  信号的寿命:信号的生成和传递之间的时间间隔。

  挂起的信号:已经生成但还未被传递的信号。

二 产生信号

  每个信号名都是以SIG开头,信号的名字都定义在signal.h中,POSIX必需的信号如下:


  信号         描述 


  SIGABRT       进程放弃(signal abort)

  SIGALRM       告警时钟(signal alarm)

  SIGBUS       访问了内存对象中的为定义部分

  SIGCHLD       子进程被终止,停止或继续(signal child)

  SIGCONT       如果进程被停止了,本信号使进程继续执行(signal continue)

  SIGFPE       算术计算中出现了被零除这样的错误

  SIGHUP       在控制终端(进程)上挂起(死亡)

  SIGILL         无效的硬件指令

  SIGINT        交互终端提示信号(通常是Ctrl-C)

  SIGKILL       终止(signal kill)

  SIGPIPE       向一个没有读程序的管道写入(signal pipe)

  SIGQUIT       交互终端终止(通常是Ctrl-l)(signal quit)

  SIGSEGV        无效的内存引用

  SIGSTOP        执行停止(signal stop)

  SIGTERM       终止(signal terminate)

  SIGTTIN       后台进程试图进行读操作(signal try to input)

  SIGTTOU        后台进程试图进行写操作(signal try to output)

  SIGURG       在套接字上有高速宽数据

  SIGUSR1        用户定义的信号1

  SIGUSR2        用户定义的信号2


  1  函数kill

#include <signal.h>

int kill(pid_t pid,    //进程ID
     int sig      //信号码
     );          //成功返回0,不成功返回-1并设置errno

  参数pid的取值:>0            kill就向那个ID表示的进程发送信号

           =0      kill就向调用程序的进程组成员发送信号

           -1       kill就向所有它有权发送信息的进程发送信号

          其它负值    就将信号发送到组ID等于|pid|的进程组中去  

  kill的实现必须检测的错误及相应的错误码:EINVAL:sig是一个无效的或不被支持的信号

                      EPERM:调用程序没有适当的权限

                      ESRCH:没有进程或进程组对应于pid

  例子:向进程1000发送SIGUSR1 

if(kill(,SIGUSR1)){

    perro("Failed to send the SIGUSR1 signal");

}

  注:查找相关进程ID的常用方法是使用getpid(获取当前进程ID),getppid(获取当前进程的父进程ID),getpgid(获取当前进程的进程组ID),或者通过保存从fork中返回的值来查找。

  2 函数raise

#include <signal.h>

int raise(int sig);  //成功,返回0,不成功如果sig是无效的,raise函数就将error设置为EINVAL

  raise函数用来向自己发送一个信号。

  例子:使进程向自己发送一个SIGUSR1信号

if(raise(SIGUSR1)!=){

  perror("Failed to raise SIGUSR1");

}

  3 函数alarm

#include <unistd.h>

unsigned alarm(unsigned seconds);

  alarm函数用来在seconds秒之后安排发送一个SIGALRM信号,alarm函数从来不报告错误。

三 对信号掩码和信号集的操作

  信号掩码:当前被阻塞的信号的集合,类型为sigset_t.

  对信号集的操作由以下五个函数组成:

#include <signal.h>

int sigaddset(sigset_t *set,int signo);        //将signo加入信号集

int sigdelset(sigset_t *set,int signo);        //将signo从信号集中删除

int sigemptyset(sigset_t *set);             //对信号集初始化,使其不包含任何信号

int sigfillset(sigset_t *set);             //对信号集初始化,使其包含所有信号

int sigismember(const sigset_t *set,int signo);   //报告signo是否在*set中,如果在,返回1,否则返回0

  例子:对信号集twosigs进行初始化,使其包含两个信号SIGINT和SIGQUIT

if((sigemptyset(&twosigs)==- || sigaddset(&twosigs,SIGINT)==- || sigaddset(&twosigs,SIGQUIT)==-)){

  perror("Failed to set up signal mask");

}

  进程可以用sigprocmask函数来检查或修改它的进程信号掩码,sigprocmask函数可以根据参数how指定的方法修改进程的信号掩码。新的信号掩码由参数set指定,而原先的信号掩码将保存到信号集oset中,声明如下:

#include <signal.h>

int signalprocmask(int how,                //用来说明信号掩码的修改方式
           const sigset_t *restrict set,    //指向一个信号集的指针,在修改中要用到这个信号集,如果为NULL,就说明不需要进行修改
           sigset_t *restrict oset        //如果不为NULL,sigprocmask会将修改之前的信号集放在*oset中返回
           );

  参数how取以下三个值中的一个:

  SIG_BLOCK:向当前被阻塞的信号中添加一个信号集

  SIG_UNBLOCK:从当前被阻塞的信号中删除一个信号集

  SIG_SETMASK:将指定的信号集设置为被阻塞的信号

  例子:将SIGINT添加到进程已经阻塞的信号集中去

sigset_t newsigset;

if((sigemptyset(&newsigset)==- || sigaddset(&newsigset,SIGINT)==-)){

  perror("Failed to initialize the signal set");

}

else if(sigprocmask(SIG_BLOCK,&newsigset,NULL)==-){

  perror("Failed to block SIGINT");

}

Linux学习笔记18——信号1的更多相关文章

  1. Linux学习笔记19——信号2

    上一节中讲到了sigprocmask函数,它的作用是检查或修改它的进程信号掩码,这一节我们主要学习捕捉与忽略信号的函数sigaction和等待信号函数. 一  sigaction函数的作用 定义在接收 ...

  2. Linux学习笔记(18) Shell编程之流程控制

    1. if语句 (1) 单分支if条件语句 格式为: # 注意条件判断式两端的空格if [ 条件判断式 ];then 程序员 fi 或者 if[ 条件判断式 ] then 程序 fi 例:判断分区使用 ...

  3. Linux学习笔记(六) 进程管理

    1.进程基础 当输入一个命令时,shell 会同时启动一个进程,这种任务与进程分离的方式是 Linux 系统上重要的概念 每个执行的任务都称为进程,在每个进程启动时,系统都会给它指定一个唯一的 ID, ...

  4. Linux 学习笔记

    Linux学习笔记 请切换web视图查看,表格比较大,方法:视图>>web板式视图 博客园不能粘贴图片吗 http://wenku.baidu.com/view/bda1c3067fd53 ...

  5. Linux 学习笔记之超详细基础linux命令 Part 11

    Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 10---------------- ...

  6. Linux 学习笔记之超详细基础linux命令 Part 9

    Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 8----------------- ...

  7. Linux 学习笔记之超详细基础linux命令 Part 8

    Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 7----------------- ...

  8. Linux 学习笔记之超详细基础linux命令 Part 1

    Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122   说明:主要是在REHL Server 6操作系统下进行的测试 --字符界面虚拟终端与图形界面之间的切 方法:[ ...

  9. Linux学习笔记-Linux系统简介

    Linux学习笔记-Linux系统简介 UNIX与Linux发展史 UNIX是父亲,Linux是儿子. UNIX发行版本 操作系统 公司 硬件平台 AIX IBM PowerPC HP-UX HP P ...

随机推荐

  1. MICROSOFT REPORT VIEWER 2012之无法加载相关的dll

    使用VS 2012开发报表, 如果是使用的微软的报表控件的话,默认是使用的MICROSOFT REPORT VIEWER 2012,本地开发基本上没问题,但是一发布服务器,就会发现坑了,微软挖坑从来就 ...

  2. 黑马程序员- IO(Input- Output)(一)

    ------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- API包: Java.io.* 缘来: java通过操作数据对象是通过流的方式来创建的 作用: ...

  3. C#调用ActiveX控件

    背景:最近项目中需要用到ActiveX控件,项目是在.Net平台下开发的.因此就直接在项目中添加了对ActiveX控件的引用,添加引用成功.在代码中实例化类的实例也没有问题,但在调用其方法或属性时总是 ...

  4. Angularjs总结(五)指令运用及常用控件的赋值操作

    1.常用指令 <div ng-controller="jsyd-controller"> <div style="float:left;width:10 ...

  5. 数据挖掘学习笔记:挖掘频繁模式、关联和相关[ZZ]

    所 谓挖掘频繁模式,关联和相关,即指在出现的数据集中找到一个经常出现的序列模式或者是一个经常出现的数据结构.就像搞CPU设计的人知道,Cache的预 取机制有流预取和指针预取,前者就是发现流模式,即发 ...

  6. 左偏树(Leftist Heap/Tree)简介及代码

    左偏树是一种常用的优先队列(堆)结构.与二叉堆相比,左偏树可以高效的实现两个堆的合并操作. 左偏树实现方便,编程复杂度低,而且有着不俗的效率表现. 它的一个常见应用就是与并查集结合使用.利用并查集确定 ...

  7. How To Read a Paper.md

    @ Titile How To Read a Paper.md @ author Keshav, 译 uuplusu   # 1. Intro    1. 读论文重要    2. 没有人教    3. ...

  8. java是通过值传递,也就是通过拷贝传递——通过方法操作不同类型的变量加深理解(勿删)

    head first java里写到“java是通过值传递的,也就是通过拷贝传递”,由此得出结论,方法无法改变调用方传入的参数.该怎么理解呢? 看例子: public class Test1 { pu ...

  9. WPF学习(一)控件的公共属性

    Visiblity控件是否可见:枚举类型:Visible表示可见.Collapsed不可见. IsEnabled:控件是否可用:bool类型. Background:背景色. FontSize:字体大 ...

  10. Hibernate报错 ** is not mapping

    使用easyui+struts+hibernate 新增加一个页面功能时,总是报错,后来发现是数据库语句,不能写表名称,而是要写映射的数据库实体类名 1.struts文件修改增加action < ...