[Linux]继续探究mysleep函数(竞态条件)

之前我们探究过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秒之后:

这里写图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/382706.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言的atoi和C++的to_string

to_stringint to string将其他型转换成字符串型atoiascii to integer是把字符串转换成整型数的一个函数 to_string #include <iostream> // std::cout #include <string> // std::string, std::to_stringint main () {std::string perfect std::to_string…

ubuntu 升级python3.5到python3.7,并升级pip3

1, 下载python3.7.tgz 文件&#xff0c;解压&#xff0c; 2. 编译安装 3. 删除 /usr/bin 目录下的 pip3, python3 4. 建立新的软连接&#xff1a; #添加python3的软链接ln -s /usr/local/python3/bin/python3.7 /usr/bin/python3#添加 pip3 的软链接ln -s /usr/local/python3/b…

[Linux]死锁

死锁是指多个进程在运行过程中因争夺资源而造成的一种僵局&#xff0c;当进程处于这种僵持状态时&#xff0c;若无外力作用&#xff0c;它们都将无法再向前推进。之前信号量的时候我们知道&#xff0c;如果多个进程等待&#xff0c;主要体现在占有锁的问题上。死锁也可以被定义…

Python安装第三方模块总结 转载的

转自 https://www.jellythink.com/archives/541

[C++]vector创建二维数组

c.resize(n);将c重置为大小为n个元素向量&#xff0c;如果n比原来的元素多&#xff0c;则多出的元素常被初始化为0//节选《面向对象的程序设计》杜茂青 int N5, M6; vector<vector<int> > Matrix(N); for(int i 0; i< Matrix.size(); i){ Matrix[i].resize(M…

[Linux]线程安全和可重入函数

线程安全&#xff1a;一个函数被称为线程安全的&#xff0c;当且仅当被多个并发进程反复调用时&#xff0c;它会一直产生正确的结果。如果一个函数不是线程安全的&#xff0c;我们就说它是线程不安全的。 重入&#xff1a;函数被不同的控制流程调用,有可能在第一次调用还没返回…

[Linux]信号量

信号量是一个计数器&#xff0c;用于为多个进程提供对共享数据对象的访问。 在信号量上只有三种操作可以进行&#xff0c;初始化、递增和增加&#xff0c;这三种操作都是原子操作。递减操作可以用于阻塞一个进程&#xff0c;增加操作用于解除阻塞一个进程。 为了获得共享资源…

Linux VIM 程序中有游离的‘\357’ ‘\274’错误

gcc date.cpp -o date -lstdc date.cpp:18:20: 错误&#xff1a;程序中有游离的‘\357’date.Showdata()&#xfffd;&#xfffd;&#xfffd;^ date.cpp:18:21: 错误&#xff1a;程序中有游离的‘\274’date.Showdata()&#xfffd;&#xfffd;&#xfffd;^ date.cpp:18:22…

[Linux]关于SIGCHLD

之前我们就学过&#xff0c;关于wait和waitpid来处理僵尸进程&#xff0c;父进程等待子进程结束后自己才退出&#xff0c;这样的方法有俩种方式&#xff0c;一种是父进程死死的等子进程退出&#xff0c;也就是使用阻塞的方式等待子进程退出&#xff0c;另一种方式是通过非阻塞的…

C语言思维导图

本人能力有限&#xff0c;知识点难免概括不全&#xff0c;如有错误欢迎指正

转载一篇关于curl的文章

转载一篇关于curl的文章 http://www.360doc.com/content/16/0107/15/18578054_526158476.shtml

[Linux]vi/vim下添加多行注释和取消注释

添加注释&#xff08;Centos&#xff09;&#xff1a; 在命令行模式下按ctrlV进入 visual block模式&#xff08;可视化模式&#xff09; 选中你需要注释的行&#xff0c;再按大写的I&#xff0c;输入//&#xff0c;最后按俩下esc即可。 如果想让前进tab个位&#xff0c;则可在…

pthread和互斥量条件变量函数意义速查表

数据类型 pthread_t 线程 互斥量和条件变量

[Linux]共享内存

共享内存是UNIX提供的进程间通信手段中速度最快的一种&#xff0c;也是最快的IPC形式。为什么是最快的呢&#xff0c;因为数据不需要在客户进程和服务器进程之间复制&#xff0c;所以是最快的一种IPC。这是虚存中由多个进程共享的一个公共内存块。 两个不同进程A、B共享内存的…

僵尸进程的产生,危害和解决方案

概念 僵死状态&#xff08;Zombies&#xff09;是一个比较特殊的状态。 当进程退出并且父进程没有读取到子进程退出的返回代码时就会产生僵尸进程。僵尸进程会以终止状态保持在进程表中&#xff0c;并且会一直在等待父进程读取退出状态代码。所以&#xff0c;只要子进程退出&…

CString string 转换

https://www.cnblogs.com/HappyEDay/p/7016162.html

[Linux]gdb调试多进程多线程例程

gdb相信学linux的同学已经比较熟悉了吧&#xff0c;它是linux下代码调试工具。我们在写c语言&#xff0c;c的代码时经常会用到&#xff0c;它有一些常用的调试命令: run&#xff08;r&#xff09;&#xff1a;运行程序&#xff0c;如果有断点在下一个断点处停止 start&#xf…

gdb调试常用命令速查(段错误调试)

编译程序时需要加上-g&#xff0c;之后才能用gdb进行调试&#xff1a;gcc -g main.c -o main gdb中命令&#xff1a; 回车键&#xff1a;重复上一命令 &#xff08;gdb&#xff09;help&#xff1a;查看命令帮助&#xff0c;具体命令查询在gdb中输入help 命令,简写h &…

C语言字符串 小记

#include "stdafx.h" #include <iostream> #include <string.h> using namespace std;int _tmain(int argc, _TCHAR* argv[]) {char str1[] "12345"; // ""括起来的字符串 会在末尾增加 \0 cout << sizeof(str1) << en…

[Linux]守护进程(精灵进程)

一、守护进程是什么 守护进程是生存期很长的一种进程&#xff0c;可以说它是7*24小时工作的。&#xff08;什么是7*24&#xff0c;一周7天&#xff0c;每天24小时&#xff0c;这不就是一年365天一直在工作嘛&#xff0c;还搞的这么诙谐&#xff0c;哈哈&#xff09;。它们常常…