转:步步LINUX C--进程间通信(二)信号
源地址:http://blog.csdn.net/jmy5945hh/article/details/7529651
linux间进程通信的方法在前一篇文章中已有详细介绍。http://blog.csdn.net/jmy5945hh/article/details/7350564
本篇详细介绍及代码测试第二种方式,即信号(Signal)。
1 信号简介
信号全称为软中断信号,主要用于进程控制。
信号是进程间通信机制中唯一的异步通信机制,可以看作是异步通知。信号机制经过POSIX实时扩展后,功能更加强大,除了基本通知功能外,还可以传递附加信息。
信号的来源包括硬件来源与软件来源。
生存周期从被创建开始,到进程接收到信号。某些情况下允许信号排队。
内核对信号的处理机制:
(1)发送信号的方法为在进程所在的进程表项的信号域设置对应于该信号的位。
(2)睡眠进程的睡眠优先级高于信号时,信号不会唤醒进程。
(3)内核处理信号的时机是进程从内核态返回用户态时或者进程要切换到低优先级睡眠态时。
(4)处理方法:退出,忽略或通过signal函数调用用户定义函数处理。
典型应用:LINUX下程序在命令行运行时,按下CTRL+C将向程序发送SIGINT信号,强制进程结束。
在命令行使用kill -l命令可以列出所有LINUX系统支持的信号

值得注意的是,信号的编号不是连续的。在SIGRTMIN之前的信号被称为不可靠信号,之后的称为可靠信号。
主要的区别在于,不可靠信号:
1/进程每次处理信号后,就将对信号的响应设置为默认动作。在某些情况下,将导致对信号的错误处理;因此,用户如果不希望这样的操作,那么就要在信号处理函数结尾再一次调用signal(),重新安装该信号。
2/不支持排队,信号可能丢失。 因此,早期unix下的不可靠信号主要指的是进程可能对信号做出错误的反应以及信号可能丢失。
2 信号的处理
注册处理函数
- #include <signal.h>
- void (*signal (int signum, void (*handler) (int) ) ) (int);
- /* 等效形式 */
- typedef void(*sighandler_t) (int);
- sighandler_t signal(int signum, sighandler_t handler);
#include <signal.h>
void (*signal (int signum, void (*handler) (int) ) ) (int);
/* 等效形式 */
typedef void(*sighandler_t) (int);
sighandler_t signal(int signum, sighandler_t handler);
成功返回之前的处理信号配置;出错返回SIG_ERR。
不能捕捉SIGKILL或者SIGSTOP信号。handler取值为SIG_IGN表示忽略,为SIG_DFL表示执行系统默认操作,为函数名时表示执行特定操作。
处理函数增强版
- #include <signal.h>
- int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
#include <signal.h>
int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
成功后返回0;出错返回-1。
- struct sigaction
- {
- void (*sa_sighandler) (int);
- void (*sa_sigaction) (int, siginfo_t *, void *);
- sigset_t sa_mask;
- int sa_flags;
- void (*sa_restorer) (void);
- }
struct sigaction
{
void (*sa_sighandler) (int);
void (*sa_sigaction) (int, siginfo_t *, void *);
sigset_t sa_mask;
int sa_flags;
void (*sa_restorer) (void);
}
sa_flags参数
| 取值 | 含义 |
| SA_NOCLDSTOP | 用于指定信号SIGCHLD。 |
| SA_NOCLDWAIT | 对信号SIGCHLD,子进程结束时,不创建僵死进程。 |
| SA_NODEFER | 处理信号时如果产生了其他信号,则先去处理其他信号。 |
| SA_NOMASK | 与SA_NODEFER相似。 |
| SA_RESETHAND | 处理完信号后,注销处理函数。 |
| SA_ONESHOT | 同SA_RESETHAND相似 |
| SA_RESTART | 信号发生时正处在程序阻塞处,则处理完信号从阻塞处返回。 |
| SA_SIGINFO | 指示sa_sigaction有效。不设置表示sa_handler有效。 |
信号集
- #include <signal.h>
- int sigemptyset( sigset_t *set );
- int sigfillset( sigset_t *set );
- int sigaddset( sigset_t *set, int signum );
- int sigdelset( sigset_t *set, int signum );
- int sigismember( const sigset_t *set, int signum );
#include <signal.h>
int sigemptyset( sigset_t *set );
int sigfillset( sigset_t *set );
int sigaddset( sigset_t *set, int signum );
int sigdelset( sigset_t *set, int signum );
int sigismember( const sigset_t *set, int signum );
前四个函数成功返回0;出错返回-1.最后一个函数为真返回1;为假返回0。
信号集用于表示由多个信号所组成的数据类型。empty置空,fill置满,add添加,del删除,member检测。
3 信号的发送
向其他进程发送信号
- #include <signal.h>
- #include <sys/types.h>
- int kill( pid_t pid, int signum );
#include <signal.h>
#include <sys/types.h>
int kill( pid_t pid, int signum );
成功后返回0;出错返回-1。发送对象可以使一个进程组。
向本进程发送信号
- #include <signal.h>
- #include <sys/types.h>
- int raise( int signum );
#include <signal.h>
#include <sys/types.h>
int raise( int signum );
成功后返回0;出错返回-1。
实时信号的发送
- #include <signal.h>
- #include <sys/types.h>
- int sigqueue( pid_t pid, int signum, const union sigval val );
#include <signal.h>
#include <sys/types.h>
int sigqueue( pid_t pid, int signum, const union sigval val );
- typedef union sigval
- {
- int sival_int;
- void *sigval_ptr; // 指向要传递的信号参数
- } sigval_t;
typedef union sigval
{
int sival_int;
void *sigval_ptr; // 指向要传递的信号参数
} sigval_t;
支持信号带有参数。成功后返回0;出错返回-1。发送对象只能是单个进程。
指定时间的发送
- #include <unistd.h>
- unsigned int alarm( unsigned int seconds );
#include <unistd.h>
unsigned int alarm( unsigned int seconds );
seconds参数指定了下一次发送SIGALRM信号的时间。如果调用前设置了闹钟,则返回闹钟剩余时间,否则返回0。
- #include <sys/time.h>
- int setitimer( int which, const struct itimerval *value, struct itimerval *oldvalue);
- struct itimerval
- {
- struct timeval it_interval;
- struct timeval it_value;
- };
#include <sys/time.h>
int setitimer( int which, const struct itimerval *value, struct itimerval *oldvalue); struct itimerval
{
struct timeval it_interval;
struct timeval it_value;
};
which参数
| 取值 | 定时器类型 | 信号 |
| ITIMER_REAL | 根据系统时间设定绝对时间。 | SIGALRM |
| ITIMER_VIRTUAL | 设定程序执行时间,用户模式下可跟踪。 | SIGVTALRM |
| TIMER_PROF | 从用户进程开始后开始计时。 | SIGPROF |
成功后返回0;出错返回-1。相比于alarm,setitimer支持三种类型的定时器。
3 信号的阻塞
- #include <signal.h>
- int sigprocmask( int how, const sigset_t *set, sigset_t *oldset );
#include <signal.h>
int sigprocmask( int how, const sigset_t *set, sigset_t *oldset );
how参数
| 取值 | 功能 |
| SIG_BLOCK | 将set信号集的信号与掩码做逻辑或运算 |
| SIG_UNBLOCK | 将set信号集的信号与掩码做逻辑减运算 |
| SIG_SETMASK | 以set信号集对信号掩码进行赋值操作 |
成功后返回0;出错返回-1。用于信号的阻塞延迟处理。
- #include <signal.h>
- int sigsuspend( const sigset_t *sigmask );
#include <signal.h>
int sigsuspend( const sigset_t *sigmask );
成功后无返回值;出错返回-1。调用此函数后进程挂起,直到接收到信号复原信号集,然后调用处理函数。
3 信号通信测试代码:
- #include <stdio.h>
- #include <signal.h>
- #include <sys/types.h>
- #include <unistd.h>
- void op(int, siginfo_t*, void*);
- int main(int argc,char**argv) {
- struct sigaction act;
- int sig;
- pid_t fpid, spid;
- if((spid =fork()) == 0) {
- int signum;
- union sigval mysigval;
- signum = 5; //信号值取5
- fpid = getppid(); //获取父进程ID
- mysigval.sival_int = 8; // 用于测试
- if(sigqueue(fpid, signum, mysigval) == -1) //子进程发送信号
- printf("send error\n");
- sleep(2);
- }
- else if(spid > 0) {
- struct sigaction act;
- int sig = 5;
- sigemptyset(&act.sa_mask);
- act.sa_sigaction=op;
- act.sa_flags=SA_SIGINFO;
- if(sigaction(sig, &act, NULL) < 0) //父进程指定对于信号的处理
- printf("install sigal error\n");
- sleep(2);
- }
- }
- void op(int signum, siginfo_t *info, void *myact) { // 信号处理函数
- printf("the int value is %d \n", info->si_int);
- }
运行结果:
- jimmy@MyPet:~/code/ipc$ gcc -g -o ipc ipc.c
- jimmy@MyPet:~/code/ipc$ ./ipc
- the int value is 8
非正常测试结果及总结
1)将子进程中的signum改为与父进程中的sig值不一致,不能进行信号通信。
2)信号通信是惟一的进程间异步通信模式。
3)可以在终端输入kill -l查看系统支持的信号。其中前半部分(<=32)是不可靠信号,后半部分是可靠信号。
4)进程对于实时信号的默认处理方式都是终止进程。
转:步步LINUX C--进程间通信(二)信号的更多相关文章
- Linux环境进程间通信(二):信号(下)
linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...
- Linux环境进程间通信(二): 信号(上)
linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...
- <转>Linux环境进程间通信(二): 信号(上)
原文链接:http://www.ibm.com/developerworks/cn/linux/l-ipc/part2/index1.html 原文如下: 一.信号及信号来源 信号本质 信号是在软件层 ...
- Linux环境进程间通信(二): 信号--转载
http://www.ibm.com/developerworks/cn/linux/l-ipc/part2/index1.html http://www.ibm.com/developerworks ...
- [转]Linux进程间通信——使用信号
转载于:http://blog.csdn.net/ljianhui/article/details/10128731 经典!!! Linux进程间通信——使用信号 一.什么是信号 用过 ...
- linux内核剖析(九)进程间通信之-信号signal
信号及信号来源 什么是信号 信号是UNIX和Linux系统响应某些条件而产生的一个事件,接收到该信号的进程会相应地采取一些行动.通常信号是由一个错误产生的.但它们还可以作为进程间通信或修改行为的一种方 ...
- 练习--LINUX进程间通信之信号SIGNAL
同样的,信号也不要太迷信可靠信号及不及靠信号,实时或非实时信号. 但必须要了解这些信号之间的差异,函数升级及参数,才能熟练运用. ~~~~~~~~~~~~~~~~ 信号本质 信号是在软件层次上对中断机 ...
- Linux进程间通信(三) - 信号
什么是信号 软中断信号(signal,又简称为信号)用来通知进程发生了异步事件.在软件层次上是对中断机制的一种模拟,在原理上,一个进程收到一个信号与处理器收到一个中断请求可以说是一样的.信号是进程间通 ...
- 【转载】Linux的进程间通信-信号量
原文:Linux的进程间通信-信号量 Linux的进程间通信-信号量 版权声明: 本文章内容在非商业使用前提下可无需授权任意转载.发布. 转载.发布请务必注明作者和其微博.微信公众号地址,以便读者询问 ...
随机推荐
- DELPHI指针的使用
DELPHI指针的使用 大家都认为,C语言之所以强大,以及其自由性,很大部分体现在其灵活的指针运用上.因此,说指针是C语言的灵魂,一点都不为过.同时,这种说法也让很多人产生误解,似乎只有C语言的指针才 ...
- day20_函数的闭包 与 装饰器
#!/usr/bin/env python # -*- coding:utf-8 -*- # # 一些文章 # https://www.cnblogs.com/Vae1242/p/6944338.ht ...
- 洛谷P1792——[国家集训队]种树
传送门:QAQQAQ 题意:$n$个点中选$m$个不相邻的点,使得这些点不相邻(1和n算相邻),求这些点的最大值 思路:这不是神仙题不是神仙题…… 刚看到这题觉得不难,好像只要贪心就可以了但贪心不知从 ...
- java编程规约一
提高开发效率,比较重视代码规范,尤其是可扩展性和可维护性,以及可读性.如果你是一个刚进公司的开发者,最好先问问前辈是否有 内部的开发规范,花点时间过一遍.即使提交代码没有review的步骤,自己心里应 ...
- java-day06
面向过程 每一个具体的步骤都亲力亲为,详细处理每一个细节 面向对象 不关心具体步骤,而是找一个已经具有该功能的人来帮我做事 特点 封装性 继承性 多态性 类 是一组相关属性和行为的集合 成员变量(属性 ...
- 初识OpenCV-Python - 005: 识别视频中的蓝色
此次主要学习了如何将BGR转成HSV,主要用到cv2.cvtColor()和cv2.inRange()函数来识别视频中的蓝色物体. code: import cv2import numpy as np ...
- springcloud系列13 config的客户端使用
config客户端的使用: 也是首先要引入依赖: <dependency> <groupId>org.springframework.cloud</groupId> ...
- 关于Windows10企业版的激活方法
今天打开Excel在使用的时候,突然弹出弹窗,说我的激活即将过期什么的,让我转到设置进行激活. 第一个想到的办法就是更换产品密钥,在网上找了不少产品密钥,密钥有效,但是需要连接企业激活什么的,因为我是 ...
- html--双飞翼布局
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- 【JZOJ3400】旅行
description 从前有一位旅者,他想要游遍天下所有的景点.这一天他来到了一个神奇的王国:在这片土地上,有n个城市,从1到n进行编号.王国中有m条道路,第i条道路连接着两个城市ai,bi,由于年 ...