54. {
55. struct sigaction act;
56. int sig;
57. pid_t pid;
58.
59. pid=getpid();
60. sig=atoi(argv[1]);
61.
62. sigemptyset(&act.sa_mask);
63. act.sa_sigaction=new_op;
64. act.sa_flags=SA_SIGINFO;
65. if(sigaction(sig,&act,NULL)<0)
66. {
67. printf("install sigal error\n");
68. }
69. while(1)
70. {
71. sleep(2);
72. printf("wait for the signal\n");
73. }
74.
75. }
76. void new_op(int signum,siginfo_t *info,void *myact)
77. {
78. printf("the int value is %d \n",info->si_int);
79. }
80.
信号发送程式:命令行第二个参数为信号值,第三个参数为接收进程ID。
#include
#include
#include
#include
main(int argc,char**argv)
{
pid_t pid;
int signum;
union sigval mysigval;
signum=atoi(argv[1]);
pid=(pid_t)atoi(argv[2]);
mysigval.sival_int=8;//不代表具体含义,只用于说明问题
if(sigqueue(pid,signum,mysigval)==-1)
printf("send error\n");
sleep(2);
}
注:实例2的两个例子侧重点在于用信号来传递信息,现在关于在linux下通过信号传递信息的实例很少,倒是Unix下有一些,但传递的基本上 都是关于传递一个整数,传递指针的我还没看到。我一直没有实现不同进程间的指针传递(实际上更有意义),也许在实现方法上存在问题吧,请实现者email 我。
实例三:信号阻塞及信号集操作
#include "signal.h"
#include "unistd.h"
static void my_op(int);
main()
{
sigset_t new_mask,old_mask,pending_mask;
struct sigaction act;
sigemptyset(&act.sa_mask);
act.sa_flags=SA_SIGINFO;
act.sa_sigaction=(void*)my_op;
if(sigaction(SIGRTMIN 10,&act,NULL))
printf("install signal SIGRTMIN 10 error\n");
sigemptyset(&new_mask);
sigaddset(&new_mask,SIGRTMIN 10);
if(sigprocmask(SIG_BLOCK, &new_mask,&old_mask))
printf("block signal SIGRTMIN 10 error\n");
sleep(10);
printf("now begin to get pending mask and unblock SIGRTMIN 10\n");
if(sigpending(&pending_mask)<0)
printf("get pending mask error\n");
if(sigismember(&pending_mask,SIGRTMIN 10))
printf("signal SIGRTMIN 10 is pending\n");
if(sigprocmask(SIG_SETMASK,&old_mask,NULL)<0)
printf("unblock signal error\n");
printf("signal unblocked\n");
sleep(10);
}
static void my_op(int signum)
{
printf("receive signal %d \n",signum);
}
编译该程式,并以后台方式运行。在另一终端向该进程发送信号(运行kill -s 42 pid,SIGRTMIN 10为42),查看结果能够看出几个关键函数的运行机制,信号集相关操作比较简单。
注:在上面几个实例中,使用了printf()函数,只是作为诊断工具,pringf()函数是不可重入的,不应在信号处理函数中使用。
结束语:
系统地对linux信号机制进行分析、总结使我受益匪浅!感谢王小乐等网友的支持!
Comments and suggestions are greatly welcome!
附录1:
用sigqueue实现的命令行信号发送程式sigqueuesend,命令行第二个参数是发送的信号值,第三个参数是接收该信号的进程ID,能够配合实例一使用:
#include
#include
#include
int main(int argc,char**argv)
{
pid_t pid;
int sig;
sig=atoi(argv[1]);
pid=atoi(argv[2]);
sigqueue(pid,sig,NULL);
sleep(2);
}
参考文献:
• linux内核源代码情景分析(上),毛德操、胡希明著,浙江大学出版社,当要验证某个结论、想法时,最好的参考资料;
• UNIX环境高级编程,作者:W.Richard Stevens,译者:尤晋元等,机械工业出版社。对信号机制的发展过程阐述的比较周详。
• signal、sigaction、kill等手册,最直接而可靠的参考资料。
• http://www.linuxjournal.com/modules.php?op=modload&name=NS-help&file=man提供了许多系统调用、库函数等的在线指南。
• http://www.opengroup.org/onlinepubs/007904975/能够在这里对许多关键函数(包括系统调用)进行查询,很好的一个网址。
• http://unix.org/whitepapers/reentrant.html对函数可重入进行了阐述。
• http://www.uccs.edu/~compsvcs/doc-cdrom/DOCS/HTML/APS33DTE/DOCU_006.HTM对实时信号给出了相当好的描述。
关于作者
郑彦兴,国防科大攻读博士学位。联系方式: mlinux@163.com.
文章整理:西部数码--专业提供域名注册、虚拟主机服务
http://www.west263.com
以上信息与文章正文是不可分割的一部分,如果您要转载本文章,请保留以上信息,谢谢!




