linux线程同步03
来源:互联网 发布:淘宝箭牌官方旗舰店 编辑:程序博客网 时间:2024/05/19 10:38
信号是向进程发送的软件通知,通知进程有事件发生。引发信号的事件发生时,信号就被生成了。进程根据信号采取行动时,信号就被传递了。信号的寿命就是信号的生成和传递之间的时间间隔。已经生成但还未被传递的信号被称为挂起的信号。在信号生成和信号传递之间可能会有相当长的时间。
signal(参数1,参数2);
参数1:我们要进行处理的信号。系统的信号我们可以再终端键入 kill -l查看(共64个)。其实这些信号时系统定义的宏。
参数2:我们处理的方式(是系统默认还是忽略还是捕获)。
一般有3中方式进行操作。
(1)eg: signal(SIGINT ,SIG_ING );
//SIG_ING 代表忽略SIGINT信号,SIGINT信号代表由InterruptKey产生,通常是CTRL +C 或者是DELETE 。发送给所有ForeGround Group的进程。
(2)eg: signal(SIGINT ,SIG_DFL );
//SIGINT信号代表由InterruptKey产生,通常是CTRL +C或者是DELETE。发送给所有ForeGroundGroup的进程。 SIG_DFL代表执行系统默认操作,其实对于大多数信号的系统默认动作时终止该进程。这与不写此处理函数是一样的。
(3)typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
void SignHandler(int iSignNo){ printf("Capture sign no:%d/n",iSignNo); }<strong><span style="color:#FF0000;">正常编译的时候不会出现恢复信号默认处理行为,默认编译,signal()调用就是POSIX行为,等同于调用sigaction,会阻塞,没有disposition.如果编译时加了额外的选项例如-std=c99,那么就是SysV的行为,调用内核的signal。</span></strong>
在signal处理机制下,还有许多特殊情况需要考虑:
1、 注册一个信号处理函数,并且处理完毕一个信号之后,是否需要重新注册,才能够捕捉下一个信号;
2、 如果信号处理函数正在处理信号,并且还没有处理完毕时,又发生了一个同类型的信号,这时该怎么处理;
3、 如果信号处理函数正在处理信号,并且还没有处理完毕时,又发生了一个不同类型的信号,这时该怎么处理;
4、 如果程序阻塞在一个系统调用(如read(...))时,发生了一个信号,这时是让系统调用返回错误再接着进入信号处理函数,还是先跳转到信号处理函数,等信号处理完毕后,系统调用再返回。
#include <signal.h>
int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
sigaction也用于注册一个信号处理函数。
参数signum为需要捕捉的信号;
参数 act是一个结构体,里面包含信号处理函数地址、处理方式等信息。
参数oldact是一个传出参数,sigaction函数调用成功后,oldact里面包含以前对signum的处理方式的信息。
如果函数调用成功,将返回0,否则返回-1
----------------------------------我是分割线---------------------------------------------
在Linux的多线程中使用信号机制,与在进程中使用信号机制有着根本的区别,可以说是完全不同。在进程环境中,对信号的处理是,先注册信号处理函数,当信号异步发生时,调用处理函数来处理信号。它完全是异步的(我们完全不知到信号会在进程的那个执行点到来!)。然而信号处理函数的实现,有着许多的限制;比如有一些函数不能在信号处理函数中调用;再比如一些函数read、recv等调用时会被异步的信号给中断(interrupt),因此我们必须对在这些函数在调用时因为信号而中断的情况进行处理(判断函数返回时 enno 是否等于 EINTR)。
所谓的同步就是 一直阻塞等待,直到屏蔽集合中的信号到来。
#include <signal.h>
sigset_t set;
其次,向信号集中加入我们想等待的信号
#include <signal.h>
int sigemptyset(sigset_t *set);//清空信号集
int sigaddset(sigset_t *set,int signo);//将某个信号加入到信号集中
int sigdelset(sigset_t *set,int signo);//删除信号集中的某个信号
int sigfillset(sigset_t *set);//包含所有已定义的信号
最后,将该信号集中的信号加入到线程信号屏蔽字(线程信号等待队列)中
#include <signal.h>int pthread_sigmask(int how, const sigset_t *set, sigset_t *oldset);
how:
#include <signal.h>int pthread_kill(pthread_t thread, int sig);
thread:给哪个线程发送信号
#include <stdio.h>#include <pthread.h>#include <signal.h>
void *threadfunc1(void *pvoid){ int signum;sigset_t sig;int signum;sigemptyset(&sig);sigaddset(&sig,SIGUSR1);pthread_sigmask(SIG_BLOCK,&sig,NULL);//设置该线程的信号屏蔽字为SIGUSR1while(1){sigwait(&sig,&signum);//睡眠等待SIGUSR1信号的到来
printf("threadfunc1 waiting is over!\n");}}void *threadfunc2(void *pvoid){ int signum;sigset_t sig;int signum;sigemptyset(&sig);sigaddset(&sig,SIGUSR1);pthread_sigmask(SIG_BLOCK,&sig,NULL);//设置该线程的信号屏蔽字为SIGUSR1while(1){sigwait(&sig,&signum);//睡眠等待SIGUSR1信号的到来printf("threadfunc2 waiting is over!\n");}}
void main(){pthread_t thread1,thread2;pthread_create(&thread1,NULL,threadfunc1,(void *)NULL); pthread_create(&thread2,NULL,threadfunc2,(void *)NULL); pthread_detach(thread1); pthread_detach(thread2); struct sigaction act; act.sa_handler=SIG_IGN; sigemptyset(&act.sa_mask); act.sa_flags=0; sigaction(SIGUSR1,&act,0);//设置信号SIGUSR1的处理方式忽略
while(1) { printf("please input select a number in (1,2)!\n"); scanf("%d",&i); if(i==1) pthread_kill(thread1,SIGUSR1);
if(i==2) pthread_kill(thread2,SIGUSR1); }}
//编译 gcc -o pthread pthread.c -lpthread
- linux线程同步03
- linux 线程 线程同步
- linux 线程同步
- Linux中的线程同步
- Linux线程同步
- Linux 下线程同步
- Linux线程同步
- Linux线程同步
- Linux线程同步
- linux线程同步机制
- Linux 线程同步
- linux线程同步
- Linux线程及同步
- linux下线程同步
- linux 线程同步
- linux线程同步问题
- Linux多线程,线程同步
- linux线程同步浅析
- 快速排序算法 原理及golang语言实现
- Sort Colors(荷兰旗问题)
- 3.1.8 面向对象编程
- C++实验3-个人所得税计算器
- Linux大杀器命令:查找所有目录下的所有文件的字符串命令
- linux线程同步03
- UIWindow简单介绍
- Linux应用服务器搭建手册—Weblogic服务域的创建与部署
- 安卓-屏幕适配的五种方式
- JVM内幕:Java虚拟机详解
- c++作业3-多分数段函数求值、定期存款利息计算器、本月有几天?
- Linux route命令
- 3.1.9 变长参数
- C++第三次实验项目-个人所得税计算器