[Linux]继续探究mysleep函数(竞态条件)
来源:互联网 发布:深圳鹏博士数据 编辑:程序博客网 时间:2024/05/16 07:00
之前我们探究过mysleep的简单用法,我们实现的代码是这样的:
#include<stdio.h>#include<signal.h>void myhandler(int sig){}unsigned int mysleep(unsigned int timeout){ struct sigaction act,oact; act.sa_handler = myhandler; sigemptyset(&act.sa_mask); act.sa_flags = 0; sigaction(SIGALRM,&act,&oact); //信号注册函数 alarm(timeout); //闹钟timeout秒后响 pause(); //挂起等待 unsigned int ret = alarm(0); //清空闹钟 sigaction(SIGALRM,&oact,NULL); return ret;}int main(){ printf("ready sleeping!\n"); mysleep(3); printf("i am waking!\n"); return 0;}
我们首先注册了捕捉信号的函数,信号为SIGALRM,然后调用了alarm函数来设置闹钟,此时pause来挂起等待,然后内核切换到别的进程运行,在timeout秒后闹钟产生SIGALRM信号,从内核态到用户态的过程中,收到用户自定义的处理函数,就在用户态处理函数,进入处理函数myhandler时,SIGALRM信号会被自动屏蔽,当处理完函数后,自动解除屏蔽,进入到内核态执行系统调用,最后切换到用户态执行主函数控制逻辑。
这里存在一个问题,比如刚刚说的alarm函数调用完成后,pause挂起等待,当把所有的逻辑都处理完成alarm返回时,进入用户态继续pause,是不是没有任何意义呢。还有一点,我们不知道pause挂起等待是在alarm函数之内还是执行后调用的,这就出现了异步情况,我们称这种现象为竞态条件。
我们如何解决这类问题呢。。我们试着将SIGALRM信号屏蔽起来,当执行完alarm函数后再自动解除屏蔽。这样就保证是在alarm函数执行到时间后挂起的状态。但是还有一种可能是当解除屏蔽之后还有可能使SIGALRM递达,因此这种方法还是有缺陷的。我们只能用原子性的操作来避免这种问题的出现。
这时又引出了一个函数:sigsuspend包含了pause的挂起等待功能,同时解决了竞态条件的问题(与exec和pause一样,没有成功的返回值)
#include <signal.h>int sigsuspend(const sigset_t *sigmask);
调用sigsuspend时,进程的信号屏蔽字由sigmask参数指定,可以通过指定sigmask来临时解除对某个信号的屏蔽,然后挂起等待,当sigsuspend返回时,进程的信号屏蔽字恢复为原来的值,如果原来对该信号是屏蔽的,从sigsuspend返回后仍然是屏蔽的。
实现代码如下:
#include<stdio.h>#include<signal.h>void myhandler(int sig){}unsigned int mysleep(unsigned int timeout){ struct sigaction act,oact; sigset_t newmask,oldmask,suspmask; act.sa_handler = myhandler; sigemptyset(&act.sa_mask); act.sa_flags = 0; sigemptyset(&newmask); sigaddset(&newmask,SIGALRM); sigprocmask(SIG_BLOCK,&newmask,&oldmask); sigaction(SIGALRM,&act,&oact); //信号注册函数 alarm(timeout); //闹钟timeout秒后响 suspmask = oldmask; sigdelset(&suspmask,SIGALRM); sigsuspend(&suspmask); //pause(); //挂起等待 unsigned int ret = alarm(0); //清空闹钟 sigaction(SIGALRM,&oact,NULL); sigprocmask(SIG_SETMASK,&oldmask,NULL); return ret;}int main(){ printf("ready sleeping!\n"); mysleep(3); printf("i am waking!\n"); return 0;}
运行结果:
3秒之后:
阅读全文
0 0
- [Linux]继续探究mysleep函数(竞态条件)
- Linux-MySleep函数实现与竟态条件
- 【Linux】mysleep函数---普通版本与规避竞态条件版本
- Linux----普通版本的mysleep与规避竞态条件的mysleep对比
- 普通版本mysleep和规避竞态条件的mysleep
- 编写规避竞态条件的mysleep
- Linux中的mysleep函数
- 信号捕获之pause函数和竟态条件(mysleep)
- 【Linux】实现睡眠函数mysleep
- mysleep与竞态条件下的mysleep
- 【Linux】sleep函数的简单实现--mysleep
- Linux信号(二)——编写自己的mysleep函数
- mysleep与sigsuspend函数
- 编写一个mysleep函数
- 实现睡眠函数mysleep
- linux mysleep实现
- Linux 信号之mysleep
- Linux下的mysleep
- 077-C++
- 建造者模式
- OpenCV 下载驿站(百度云盘下载,同步更新)
- windows下安装redis
- iPhone安装OpenSSH和远程连接教程
- [Linux]继续探究mysleep函数(竞态条件)
- GitLab工具SourceTree使用文档
- PADS覆铜管理中灌注(Flood)和填充(Hatch)有什么区别?
- android studio git使用总结 (二): 高级篇 分支管理
- js 事件流,事件处理程序,事件对象
- eclipse自动生成的get set方法 自动加上文本注释,并且注释内容包含字段中我们加的文档注释 .
- CMFCPropertyGridCtrl 的简单教程
- 078-C++
- SonarQube Scanner的配置与使用简介