新起点
睡眠 (系统调用)
2020-08-03 05:30:22

计算机程序(或进程、任务或线程等)进入不活跃状态并持续一段时间,称为“睡眠”。当引发程序睡眠的代码所设定的内部计时器归零,或是此程序收到唤醒信号或中断时,程序将恢复执行。

睡眠中的程序也有可能不经唤醒而直接被终止。

睡眠指令通常需要输入一个参数,以说明程序睡眠至少多长时间后才恢复执行。这一时间参数的单位通常是秒,有些操作系统可以提供更高精度,例如以毫秒或微秒作为单位。

Windows 操作系统中,Sleep()函数仅有一个参数,即以毫秒计的睡眠时间。Sleep()函数包含于 kernel32.dll 中,但是原生操作系统的批处理文件却并不支持这一指令。安装诸如Windows 2003 Resource Kit等运行环境后,用户便可使用这一指令。

在类Unix操作系统中,sleep()函数使用以秒做单位的无符号整数作为输入参数。如果需要更精确的控制睡眠时间,可以使用 nanosleep()函数。

Windows 操作系统中:

Sleep(2*1000);  // 睡眠2秒

类Unix操作系统中:

sleep(2);       // 睡眠2秒

底层功能

及其他类似的系统指令会被中断信号中断,相应程序也会停止睡眠。而 库函数却不尽相同,在一些老一点的操作系统中,它是通过调用 系统指令实现的,因而必须通过送信号的方式才能运作。

在Windows中,由于除了终止运行信号外没有专门的信号可以用于打断,睡眠函数是不可中断的。不过,Windows提供了 SleepEX 函数来实现通过 APC 调用控制睡眠进程的唤醒。

严格说来,一个线程也可以由于其他线程终止运行或抛出异常被唤醒。

一些系统程序在事件轮询过程中从不被终止,只是在每个周期开始前进入睡眠状态,等待着某些事件的发生唤醒它们。当收到事件之后,它们就被唤醒去处理这些事件,然后进入下一个周期的等待。

还有一些程序周期性的从睡眠中恢复运行并收集事件。当运行恢复,程序拉取事件列表和状态变更情况,然后针对睡眠时发生的事件做相应处理,完成后再次进入睡眠,等待下一个时隙。这类事件被称作心跳事件,或“持久连接”信号,可由上述类型的周期性程序发送。

sleep()函数还在程序执行节奏需要放缓的时候被调用。例如长时间高负荷运行程序时应有包含睡眠函数的代码来减轻硬件过热带来的麻烦,或是解决遗留代码中不为人知的奇妙程序错误。相比于使用仿真器实现的周期平衡的程序,睡眠-运行周期发生的程序的缺点在于如果切换速度不够快,交互式程序就会有明显的延迟,如果太快,唤醒时间就太短以至于无法完成全部工作。

不可中断睡眠指的是在此状态的程序无法对外界信号做出任何立即的响应,只有在为它提供足够的正在等待的资源或是睡眠时间结束它才会开始运行。这类睡眠常见于硬件驱动程序等待磁盘或网络输入输出结束。当进程处于不可中断睡眠状态,睡眠期间的信号会不断堆积到队列里面去。

在类Unix操作系统中,使用指令ps -l可以列出当前终端的全部进程,其中标识进程状态的列使用代码 D 表示不可中断睡眠。这类进程连SIGKILL这样的无条件终止都没办法结束进程,结束他们最简单的方法就是重新启动系统。

网站公告: