调用sleep时操作系统中发生了什么?

相信不管你使用的是什么语言,或多或少都会调用过sleep函数,那么你知道当调用这个函数时操作系统中发生了什么吗?今天就来简单聊聊这个问题,这又是一个软件与硬件配合的经典案例。

假如我们没有操作系统,那么sleep函数的实现可能就是忙等,busy wait,即,简单的在一个for循环中消耗CPU,但有操作系统的帮助我们大可不必浪费宝贵的CPU资源。

大部分操作系统都提供了“sleep”系统调用,当我们在用户态不管用什么语言调用sleep时最终都会调用到该系统调用,系统调用后操作系统开始运行,此时:

1, 操作系统暂停该进程(线程)的执行并改变其运行状态,比如将其设置为等待状态

2,操作系统为该进程(线程)创建一个计时器,操作系统是怎样知道时间这个概念的呢?实际上很简单,假设你对时间一无所知,但我告诉你,你旁边有一条可爱的小狗,它会一秒钟汪汪汪一次,这样每当听到小狗汪汪汪时你就知道时间又过去了一秒钟,同时在纸上记下来,这样你就知道时间了。

回到我们这里,你就相当于操作系统,小狗就好比计算机系统中硬件-定时器,timer,定时器会以固定频率产生中断信号,发出中断信号后操作系统开始接管计算机系统,并开始处理一些和时间相关事情,比如检查该当前进程(线程)的时间片是否用尽、其它等待的线程是否需要唤醒等等。

3,操作系统检测到该进程(线程)定时器时间已到,将该进程(线程)从等待状态转为可执行状态,注意此时该进程(线程)可能不会立即执行,即使该进程(线程)已经位于就绪状态也要等待,因为此时系统中可能有更高优先级的进程,又或者正在运行的进程其时间片还未用完。

所以我们可以看到,假设调用sleep给定的参数是1s,那么你的进程并不会精确暂停1s然后再运行,从调用sleep到再次运行的时间一定不少于1s,也就是可能会稍多于1s,但一定不会少于1s。

在Linux系统中与sleep相关的系统调用最常见的是nanosleep,假设你有一段这样的C语言代码:

#include <unistd.h>void main() {    sleep(1);}

编译后生成的可执行程序叫做test,那么使用Linux下的strace命令运行该程序就会得到:

 

$ strace test...nanosleep({tv_sec=1, tv_nsec=0}, 0

strace命令会把程序调用的所有系统调用都显示出来,可以看到该程序最终调用nanosleep系统调用,接下来我们看看该系统调用的作用是什么,官方文档:

nanosleep会暂停当前线程的执行直到经过了参数指定的时间,或者出现某个signal,该signal触发了该线程的信号handler又或者该信号终止了该进程。

大部分用户态语言调用sleep时操作系统内部都是这样实现的。

好啦,这篇就到这里,希望对大家理解操作系统有所帮助。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

技术宅chat

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值