windows 下的信号机制初探

王朝system·作者佚名  2006-01-10
窄屏简体版  字體: |||超大  

首先牢骚牢骚啊,虽然没什么质量,然而使用文章的人是不是应该将作者名也标明呢?

在windows下,信号机制简单来说是通过工作线程实现的,该线程运行于相对优先级THREAD_PRIORITY_HIGHEST,当信号产生时,windows生成该线程执行信号处理逻辑,由于该线程优先级通常主线程,也高于用户自己显式创建的任何线程,windows线程调度逻辑将阻塞其余线程的执行,直到信号处理完毕工作线程退出.

以下是测试代码

#include "stdafx.h"

#include <signal.h>

#include <windows.h>

using namespace std;

int j = 1;

void OnCtrlC(int){

cout << "ctrl + c" << endl;

cout << ::GetCurrentThreadId() << endl;

cout <<::GetThreadPriority(::GetCurrentThread()) << endl;

signal(SIGINT,OnCtrlC);

j = 0;

}

int _tmain(int argc, _TCHAR* argv[])

{

signal(SIGINT,OnCtrlC);

int i = 0;

//::SetThreadPriority(GetCurrentThread(),THREAD_PRIORITY_TIME_CRITICAL);

while(true){

Sleep(2000);

cout << ++ i << endl;

cout << ::GetCurrentThreadId() << endl;

cout <<::GetThreadPriority(::GetCurrentThread()) << endl;

// if(j==0)

// break;

}

return 0;

}

该代码可以进行两种测试

第一种:如上,该程序运行时按下Ctrl + C后将引起 OnCtrlC函数执行,这种执行可在任何时候发生,甚至在主线程cout << 100 << endl;中仅仅输出了一个10,然后执行OnCtrlC,在然后将剩下的一个0输出.程序输出表明,OnCtrlC输出的threadid 与 main输出的threadid不同,并且OnCtrlC输出的thredid不断变化,说明改函数的线程每次都是重新创建的,OnCtrlC 调用 GetThreadPriority输出为 2,正是 THREAD_PRIORITY_HIGHEST,而main输出0,为 THREAD_PRIORITY_NORMAL.

第二种,注释掉Sleep(2000),放开main中SetThreadPriority调用,放开

if(j==0)

break;

然后运行,这次将主线程优先级调到15,高于OnCtrlC的2,因此在程序运行中按Ctrl + C将会发现程序什么反映也没有,因为主线程的高优先级阻止OnCtrlC的执行,这也是为什么放开

if(j==0)

break;

的原因,假如不放开,高速的循环代码将在你注意到OnCtrlC调用之前滚屏,既然优先级高于OnCtrlC怎么又可能OnCtrlC会被调用呢?这是由于windows动态提高线程优先级机制的作用,简单来说,就是windows注意到一个线程在3--4秒一直渴望被调度时,将被暂时将优先级提高到15,这样与main优先级相等,大家不分彼此平等竞争,在程序中产生这种情况的办法是持续不断按Ctrl + C大约3秒,程序就会退出.

tip:

只所以注释掉Sleep(2000),因为Sleep函数会将自己的Cpu时间分给其他线程.

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航