LinuxSir.cn,穿越时空的Linuxsir!

 找回密码
 注册
搜索
热搜: shell linux mysql
查看: 677|回复: 5

为什么这里直接跳过了!!

[复制链接]
发表于 2004-1-15 13:26:05 | 显示全部楼层 |阅读模式
#include <unistd.h>
#include <signal.h>
#include <stdio.h>
int main(void)
{
  sigset_t    intmask;
  sigemptyset(&intmask);
  sigaddset(&intmask,SIGALRM);
  raise(SIGALRM);
  sigprocmask(SIG_BLOCK,&intmask,NULL);
  sleep (5);
  sigprocmask(SIG_UNBLOCK,&intmask,NULL);
  pause();
  exit(0);
}
这个程序并没有sleep 5 second .
怎么能产生一个信号而继续执行下面的程序,能不能先把这个信号储存起来,要的时候再释放出来?:help :thanks
发表于 2004-1-16 11:44:09 | 显示全部楼层
这是因为你没有安装SIGALRM的处理函数吧,进程就直接退出了
发表于 2004-1-16 13:28:10 | 显示全部楼层
系统对SIGALRM的默认处理方法是终止进程
你自己写个处理函数就可以了
 楼主| 发表于 2004-1-16 15:14:40 | 显示全部楼层
#include <unistd.h>
#include <signal.h>
#include <stdio.h>
int main(void)
{
sigset_t intmask;
sigemptyset(&intmask);
sigaddset(&intmask,SIGALRM);
raise(SIGALRM);
sigprocmask(SIG_BLOCK,&intmask,NULL);
signal(SIGALRM,SIG_IGN); /* ignore the SIGALRM */
sleep (5);
sigprocmask(SIG_UNBLOCK,&intmask,NULL);
pause();
exit(0);
}
上面已忽略了sigalrm怎么还是同样的结果?
 楼主| 发表于 2004-1-16 15:39:37 | 显示全部楼层
觉得例子太少了。我贴些例程在这里,方便大家学习!!
come from:
http://www-900.ibm.com/developer ... /part2/index2.shtml


  1. linux下的信号应用并没有想象的那么恐怖,程序员所要做的最多只有三件事情:

  2.    1. 安装信号(推荐使用sigaction());
  3.    2. 实现三参数信号处理函数,handler(int signal,struct siginfo *info, void *);
  4.    3. 发送信号,推荐使用sigqueue()。

  5. 实际上,对有些信号来说,只要安装信号就足够了(信号处理方式采用缺省或忽略)。其他可能要做的无非是与信号集相关的几种操作。

  6. 实例一:信号发送及处理
  7. 实现一个信号接收程序sigreceive(其中信号安装由sigaction())。

  8. #include <signal.h>
  9. #include <sys/types.h>
  10. #include <unistd.h>
  11. void new_op(int,siginfo_t*,void*);
  12. int main(int argc,char**argv)
  13. {
  14.         struct sigaction act;       
  15.         int sig;
  16.         sig=atoi(argv[1]);
  17.        
  18.         sigemptyset(&act.sa_mask);
  19.         act.sa_flags=SA_SIGINFO;
  20.         act.sa_sigaction=new_op;
  21.        
  22.         if(sigaction(sig,&act,NULL) < 0)
  23.         {
  24.                 printf("install sigal error\n");
  25.         }
  26.        
  27.         while(1)
  28.         {
  29.                 sleep(2);
  30.                 printf("wait for the signal\n");
  31.         }
  32. }
  33. void new_op(int signum,siginfo_t *info,void *myact)
  34. {
  35.         printf("receive signal %d", signum);
  36.         sleep(5);
  37. }

  38. 说明,命令行参数为信号值,后台运行sigreceive signo &,可获得该进程的ID,假设为pid,然后再另一终端上运行kill -s signo pid验证信号的发送接收及处理。同时,可验证信号的排队问题。
  39. 注:可以用sigqueue实现一个命令行信号发送程序sigqueuesend,见附录1。

  40. 实例二:信号传递附加信息
  41. 主要包括两个实例:

  42.    1. 向进程本身发送信号,并传递指针参数;

  43. #include <signal.h>
  44. #include <sys/types.h>
  45. #include <unistd.h>
  46. void new_op(int,siginfo_t*,void*);
  47. int main(int argc,char**argv)
  48. {
  49.         struct sigaction act;       
  50.         union sigval mysigval;
  51.         int i;
  52.         int sig;
  53.         pid_t pid;               
  54.         char data[10];
  55.         memset(data,0,sizeof(data));
  56.         for(i=0;i < 5;i++)
  57.                 data[i]='2';
  58.         mysigval.sival_ptr=data;
  59.        
  60.         sig=atoi(argv[1]);
  61.         pid=getpid();
  62.        
  63.         sigemptyset(&act.sa_mask);
  64.         act.sa_sigaction=new_op;//三参数信号处理函数
  65.         act.sa_flags=SA_SIGINFO;//信息传递开关
  66.         if(sigaction(sig,&act,NULL) < 0)
  67.         {
  68.                 printf("install sigal error\n");
  69.         }
  70.         while(1)
  71.         {
  72.                 sleep(2);
  73.                 printf("wait for the signal\n");
  74.                 sigqueue(pid,sig,mysigval);//向本进程发送信号,并传递附加信息
  75.         }

  76. }

  77. void new_op(int signum,siginfo_t *info,void *myact)//三参数信号处理函数的实现
  78. {
  79.         int i;
  80.         for(i=0;i<10;i++)
  81.         {
  82.                 printf("%c\n ",(*( (char*)((*info).si_ptr)+i)));
  83.         }
  84.         printf("handle signal %d over;",signum);
  85. }


  86.       这个例子中,信号实现了附加信息的传递,信号究竟如何对这些信息进行处理则取决于具体的应用。
  87.    2. 2、 不同进程间传递整型参数:把1中的信号发送和接收放在两个程序中,并且在发送过程中传递整型参数。
  88.       信号接收程序:

  89. #include <signal.h>
  90. #include <sys/types.h>
  91. #include <unistd.h>
  92. void new_op(int,siginfo_t*,void*);
  93. int main(int argc,char**argv)
  94. {
  95.         struct sigaction act;
  96.         int sig;
  97.         pid_t pid;               
  98.        
  99.         pid=getpid();
  100.         sig=atoi(argv[1]);       
  101.        
  102.         sigemptyset(&act.sa_mask);
  103.         act.sa_sigaction=new_op;
  104.         act.sa_flags=SA_SIGINFO;
  105.         if(sigaction(sig,&act,NULL)<0)
  106.         {
  107.                 printf("install sigal error\n");
  108.         }
  109.         while(1)
  110.         {
  111.                 sleep(2);
  112.                 printf("wait for the signal\n");
  113.         }

  114. }
  115. void new_op(int signum,siginfo_t *info,void *myact)
  116. {
  117.         printf("the int value is %d \n",info->si_int);
  118. }


  119.       信号发送程序:命令行第二个参数为信号值,第三个参数为接收进程ID。

  120. #include <signal.h>
  121. #include <sys/time.h>
  122. #include <unistd.h>
  123. #include <sys/types.h>
  124. main(int argc,char**argv)
  125. {
  126.         pid_t pid;
  127.         int signum;
  128.         union sigval mysigval;

  129.         signum=atoi(argv[1]);
  130.         pid=(pid_t)atoi(argv[2]);
  131.         mysigval.sival_int=8;//不代表具体含义,只用于说明问题

  132.         if(sigqueue(pid,signum,mysigval)==-1)
  133.                 printf("send error\n");
  134.         sleep(2);
  135. }


  136.       注:实例2的两个例子侧重点在于用信号来传递信息,目前关于在linux下通过信号传递信息的实例非常少,倒是Unix下有一些,但传递的基本上都是关于传递一个整数,传递指针的我还没看到。我一直没有实现不同进程间的指针传递(实际上更有意义),也许在实现方法上存在问题吧,请实现者email我。

  137. 实例三:信号阻塞及信号集操作

  138. #include "signal.h"
  139. #include "unistd.h"
  140. static void my_op(int);
  141. main()
  142. {
  143.         sigset_t new_mask,old_mask,pending_mask;
  144.         struct sigaction act;

  145.         sigemptyset(&act.sa_mask);
  146.         act.sa_flags=SA_SIGINFO;
  147.         act.sa_sigaction=(void*)my_op;
  148.         if(sigaction(SIGRTMIN+10,&act,NULL))
  149.                 printf("install signal SIGRTMIN+10 error\n");

  150.         sigemptyset(&new_mask);
  151.         sigaddset(&new_mask,SIGRTMIN+10);
  152.         if(sigprocmask(SIG_BLOCK, &new_mask,&old_mask))
  153.                 printf("block signal SIGRTMIN+10 error\n");

  154.         sleep(10);       
  155.         printf("now begin to get pending mask and unblock SIGRTMIN+10\n");
  156.         if(sigpending(&pending_mask)<0)
  157.                 printf("get pending mask error\n");
  158.         if(sigismember(&pending_mask,SIGRTMIN+10))
  159.                 printf("signal SIGRTMIN+10 is pending\n");

  160.         if(sigprocmask(SIG_SETMASK,&old_mask,NULL)<0)
  161.                 printf("unblock signal error\n");
  162.         printf("signal unblocked\n");

  163.         sleep(10);
  164. }
  165. static void my_op(int signum)
  166. {
  167.         printf("receive signal %d \n",signum);
  168. }


  169. 编译该程序,并以后台方式运行。在另一终端向该进程发送信号(运行kill -s 42 pid,SIGRTMIN+10为42),查看结果可以看出几个关键函数的运行机制,信号集相关操作比较简单。

  170. 注:在上面几个实例中,使用了printf()函数,只是作为诊断工具,pringf()函数是不可重入的,不应在信号处理函数中使用。

  171. 结束语:

  172. 系统地对linux信号机制进行分析、总结使我受益匪浅!感谢王小乐等网友的支持!
  173. Comments and suggestions are greatly welcome!

  174. 附录1:

  175. 用sigqueue实现的命令行信号发送程序sigqueuesend,命令行第二个参数是发送的信号值,第三个参数是接收该信号的进程ID,可以配合实例一使用:

  176. #include <signal.h>
  177. #include <sys/types.h>
  178. #include <unistd.h>
  179. int main(int argc,char**argv)
  180. {
  181.         pid_t pid;
  182.         int sig;
  183.         sig=atoi(argv[1]);
  184.         pid=atoi(argv[2]);
  185.         sigqueue(pid,sig,NULL);
  186.         sleep(2);
  187. }

复制代码

参考文献:

    * linux内核源代码情景分析(上),毛德操、胡希明著,浙江大学出版社,当要验证某个结论、想法时,最好的参考资料;
    * UNIX环境高级编程,作者:W.Richard Stevens,译者:尤晋元等,机械工业出版社。对信号机制的发展过程阐述的比较详细。
    * signal、sigaction、kill等手册,最直接而可靠的参考资料。
    * http://www.linuxjournal.com/modu ... 的在线指南。
    * http://www.opengroup.org/onlinep ... 的一个网址。
    * http://unix.org/whitepapers/reen ... 进行了阐述。
    * http://www.uccs.edu/~compsvcs/do ... 当好的描述。
发表于 2004-1-16 23:18:26 | 显示全部楼层
signal()要在raise()之前。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

快速回复 返回顶部 返回列表