1:signal 函数

  原型: sighandler_t signal(int signum, sighandler_t handler)      typedef void (*sighandler_t)(int);

  描述:signal函数用来在进程中指定当一个信号到达进程后该做什么处理,主要的两种方式有忽略某些信号,(监听到SIGTERM/SIGINT)退出前的打扫工作。信号处理函数的handler有两个默认值,分别是SIG_IGNSIG_DFL,表示忽略和默认行为。而且signal函数是阻塞的,比如当进程正在执行SIGUSR1信号的处理函数,此时又来一个SIGUSR1信号,signal会等到当前信号处理函数处理完后才继续处理后来的SIGUSR1,不管后来的多少个SIGUSR1信号,统一看做一个来处理。还有SIGKILLSIGSTOP这两个信号是signal函数捕捉不到的。

  注意:signal函数的行为与linux的版本有关(因为signal属于系统调用),所以移植性不好,可用sigaction函数来替代(sigaction是标准POSIX函数),而且sigaction还扩展了signal函数的功能。

2:sigaction函数

  原型:int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

  参数:

    signum:信号值

    act:信号的处理参数

    oldact:保存信号上次安装时的处理参数(备份的作用)

  返回值:0(success),-1(error)

  1. struct sigaction {
  2. void (*sa_handler)(int); //信号处理函数
  3. void (*sa_sigaction)(int, siginfo_t *, void *);
  4. sigset_t sa_mask; //信号屏蔽集
  5. int sa_flags;
  6. void (*sa_restorer)(void);// 已废弃
  7. };

  要点:

    信号阻塞:同signal函数类似,当正处于某个信号的处理函数中时,这个信号再次到达会被阻塞,待信号处理函数完成之后再处理。

      sa_mask :信号屏蔽集,所谓屏蔽并不是忽略,屏蔽的时间段是在信号处理函数执行期间,一旦处理函数执行完毕将会重新唤醒此信号。

    sa_flags :SA_RESTART:表示如果一个信号到来时进程正在执行某个系统调用(read/write等),执行完信号处理函数后返回系统调用,此系统调用会重新调用而不会出错。

实例1(sa_mask的使用):

  1. #include <errno.h>
  2. #include <string.h>
  3.  
  4. static void signal_handler(int sig)
  5. {
  6. printf("Got SIGUSR1\n");
  7. printf("Processing SIGUSR1...\n");
  8. sleep();
  9. printf("sleep over\n");
  10. }
  11.  
  12. int main(int argc, char** argv)
  13. {
  14. char buf[];
  15. int n = ;
  16. printf("PID:%d\n", getpid());
  17. struct sigaction act, oldact;
  18. act.sa_handler = signal_handler;
  19. sigemptyset(&act.sa_mask);
  20. sigaddset(&act.sa_mask, SIGINT);//把SIGINT信号加入屏蔽集
  21. //act.sa_flags = SA_RESTART;
  22. if(sigaction(SIGUSR1, &act, &oldact) < )
  23. {
  24. perror("sigaction error");
  25. return -;
  26. }
  27. /*
  28. for(;;){
  29. memset(buf, 0, 20);
  30. if((n = read(0, buf, 20)) < 0){
  31. perror("read:");
  32. }
  33. else{
  34. buf[n] = '\0';
  35. printf("%d type got, string:%s\n", n, buf);
  36. }
  37. }
  38. */
  39. sleep();
  40. return ;
  41. }

结果:

实例2(sa_flags=SA_RESTART):

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <signal.h>
  4. #include <errno.h>
  5. #include <string.h>
  6.  
  7. static void signal_handler(int sig)
  8. {
  9. printf("Got SIGUSR1\n");
  10. printf("Processing SIGUSR1...\n");
  11. sleep();
  12. printf("sleep over\n");
  13. }
  14.  
  15. int main(int argc, char** argv)
  16. {
  17. char buf[];
  18. int n = ;
  19. printf("PID:%d\n", getpid());
  20. struct sigaction act, oldact;
  21. act.sa_handler = signal_handler;
  22. sigemptyset(&act.sa_mask);
  23. sigaddset(&act.sa_mask, SIGINT);
  24. act.sa_flags = SA_RESTART;//设置SA_RESTART flag
  25. if(sigaction(SIGUSR1, &act, &oldact) < )
  26. {
  27. perror("sigaction error");
  28. return -;
  29. }
  30.  
  31. memset(buf, , );
  32. if((n = read(STDIN_FILENO, buf, )) < ){//行缓冲阻塞
  33. perror("read:");
  34. }
  35. else{
  36. buf[n] = '\0';
  37. printf("%d type got, string:%s\n", n, buf);
  38. }
  39.  
  40. return ;
  41. }

结果:

将 act.sa_flags = SA_RESTART; 这行代码屏蔽之后:

可以看到,SA_RESTART参数确实保证了被中断的系统调用在中断之后恢复之前的状态继续执行(系统调用中断出错应该报告Interrupted system call,不知我这里为何是core dumped?)。

signal和sigaction 分析的更多相关文章

  1. Linux 信号:signal 与 sigaction

    0.Linux下查看支持的信号列表: france@Ubuntux64:~$ kill -l ) SIGHUP ) SIGINT ) SIGQUIT ) SIGILL ) SIGTRAP ) SIGA ...

  2. Linux进程间通信(一): 信号 signal()、sigaction()

    一.什么是信号 用过Windows的我们都知道,当我们无法正常结束一个程序时,可以用任务管理器强制结束这个进程,但这其实是怎么实现的呢?同样的功能在Linux上是通过生成信号和捕获信号来实现的,运行中 ...

  3. signal() 和 sigaction()

    [摘自<Linux/Unix系统编程手册>] Unix系统提供了两种方式来改变信号处置:signal() 和 sigaction(). signal() 的行为在不同Unix实现间存在差异 ...

  4. 信号处理signal、sigaction、pause、信号嵌套处理、不可重入函数

    信号的捕捉和处理 主要由signal和sigaction函数来完成.还有一个函数pause,它可用来响应任何信号,不过不做任何处理. 1.signal函数 typedef void (*sighand ...

  5. APUE学习笔记——10信号——信号接口函数 signal 和 sigaction

    signal函数     signal函数是早起Unix系统的信号接口,早期系统中提供不可靠的信号机制.在后来的分支中,部分系统使用原来的不可靠机制定义signal函数,如 Solaris 10 .而 ...

  6. linux c 之signal 和sigaction区别

    http://blog.csdn.net/muge0913/article/details/7331129 要对一个信号进行处理,就需要给出此信号发生时系统所调用的处理函数.可以对一个特定的信号(除去 ...

  7. linux 信号signal和sigaction理解

    今天看到unp时发现之前对signal到理解实在浅显,今天拿来单独学习讨论下. signal,此函数相对简单一些,给定一个信号,给出信号处理函数则可,当然,函数简单,其功能也相对简单许多,简单给出个函 ...

  8. Linux信号(signal) 机制分析

    Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...

  9. Linux信号(signal) 机制分析(转)

    [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核对于信号的处理流程包括信号的触发/注册/执 ...

随机推荐

  1. PHP性能优化 -理论篇

    什么情况下,遇到了PHP性能问题?    1 PHP语法使用的不恰当    2 使用PHP语言做不了它不擅长做的事    3 用php语言连接的服务不给力    4 PHP自身的短板    5 我也不 ...

  2. [朴孝敏][Gold]

    歌词来源:http://music.163.com/#/song?id=406924220 作曲 : Ryan S. Jhun/David Quinones/Edwin Menjivar/Mateo ...

  3. 《Learning scikit-learn Machine Learning in Python》chapter1

    前言 由于实验原因,准备入坑 python 机器学习,而 python 机器学习常用的包就是 scikit-learn ,准备先了解一下这个工具.在这里搜了有 scikit-learn 关键字的书,找 ...

  4. pat甲级1002

    1002. A+B for Polynomials (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue T ...

  5. 福大软工1816:Beta(1/7)

    Beta 冲刺 (1/7) 队名:第三视角 组长博客链接 本次作业链接 团队部分 团队燃尽图 工作情况汇报 张扬(组长) 过去两天完成了哪些任务 文字/口头描述 答辩 组织会议 复习课本 展示GitH ...

  6. Web站点性能-微观手段

    文章:网站性能优化 百度百科:高性能Web站点 文章:构建高性能WEB站点之 吞吐率.吞吐量.TPS.性能测试

  7. lintcode-223-回文链表

    223-回文链表 设计一种方式检查一个链表是否为回文链表. 样例 1->2->1 就是一个回文链表. 挑战 O(n)的时间和O(1)的额外空间. 标签 链表 思路 找到链表中点后,翻转链表 ...

  8. Java 异常注意事项

    异常的注意事项:   1,子类在覆盖父类方法时,父类的方法如果抛出了异常, 那么子类的方法只能抛出父类的异常或者该异常的子类.   2,如果父类抛出多个异常,那么子类只能抛出父类异常的子集.     ...

  9. RequestMappingHandlerMapping 详解

    我们先理简单梳理一个关系 关系梳理 spring ioc 是spring的核心,用来管理spring bean的生命周期 MVC 是一种使用 MVC(Model View Controller 模型- ...

  10. rabbitmqctl 的常用命令

    # 查看服务器的状态 rabbitmqctl status   # 查看环境变量 rabbitmqctl environment   # 停止rabbitmq的应用 rabbitmqctl stop_ ...