现在的位置: 首页 > 综合 > 正文

Windows中子线程不能触发定时器的问题分析

2013年03月08日 ⁄ 综合 ⁄ 共 649字 ⁄ 字号 评论关闭

对窗口来说,都会有一个消息循环线程(此消息线程即UI线程)执行各类任务,一般情况此线程大部分时间都处于空闲状态,由消息泵等待消息触发各类操作(如界面刷新、定时器响应等),除非窗口退出,不然不会退出此线程。

在窗口线程或主线程中使用定时器由于有消息泵等待定时器消息,所以不会存在不响应定时器的情况。但子线程在创建时一般不需要消息泵,所以按顺序执行完后直接退出,就算是有定时器需要触发,子线程也不知道,所以等定时器Timeout时,并不会达到触发的目的。

解决办法

1、所有触发定时器的操作转到主线程执行

此方法是碰到此问题时首先想到的解决方法,也是简单可行的,但某些情况可能就要改变定时器的触发方式,在修改时会比较花时间。最关键的是:此方法是在逃避问题而不是迎接问题。

2、在子线程加入消息泵

问题的本质也清楚了,那在子线程中加入消息泵即可完美解决此类问题。需要在有触发定时器的线程中加入PeekMsg()函数(代码如下),如果响应完后退出线程,直接在响应过程后退出while循环可以达到目的。

void PeekMsg()
{
   MSG msg;
   PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE);
   while(1)
   {
      if(GetMessage(&msg, NULL, 0, 0))
      {
         switch(msg.message)
         case WM_TIMER:
         {
              ::DispatchMessage(&msg);
              break;
         }    
      }
   }
}

经验证,以上方法是正确的,但需要根据实际问题进行一些调整完善。

抱歉!评论已关闭.