VC语言

本类阅读TOP10

·VC++ 学习笔记(二)
·用Visual C++打造IE浏览器(1)
·每个开发人员现在应该下载的十种必备工具
·教你用VC6做QQ对对碰外挂程序
·Netmsg 局域网聊天程序
·Windows消息大全
·VC++下使用ADO编写数据库程序
·VC++学习笔记(四)
·非法探取密码的原理及其防范
·怎样在VC++中访问、修改注册表

分类导航
VC语言Delphi
VB语言ASP
PerlJava
Script数据库
其他语言游戏开发
文件格式网站制作
软件工程.NET开发
具有Reset功能的多线程同步队列 - 4

作者:未知 来源:月光软件站 加入时间:2005-2-28 月光软件站

BOOL CTreadSafeMsgQueue::PostMsg(const MsgItem Msg, int WaitTime)

{

         if (!m_bInitedOK) return FALSE;

 

         // 防止在reset期间导致Semaphore计数不正确

         while (m_bStop)

         {

                  TRACE("Thread %d Sleep\n", GetCurrentThreadId());

                  Sleep(SLEEP_TIME);

         }

 

         // 进入同步操作状态

         m_WritingThreadNum++;

        

         if (WaitForSingleObject(m_S_Producer, WaitTime) == WAIT_OBJECT_0)

         {

                   if (WaitForSingleObject(m_E_Queue, WaitTime) == WAIT_OBJECT_0)

                  {

                            // OK now, post message then

                            m_Queue[m_HeaderToWrite++] = Msg;

                            if (m_HeaderToWrite >= MAX_QUE_SIZE) m_HeaderToWrite = 0;

                            TRACE("Post message *** %d\n", Msg.MsgID);

 

                            ReleaseSemaphore(m_S_Consumer, 1, NULL);

                            SetEvent(m_E_Queue);

 

                            m_WritingThreadNum--;

                            return TRUE;

                   }

                   else // wait event time out

                   {

                            // not post message so release producer

                            ReleaseSemaphore(m_S_Producer, 1, NULL);

                            SetEvent(m_E_Queue);

 

                            m_WritingThreadNum--;

                            return FALSE;

                   }

         }

         else // wait semaphore time out

         {

                  m_WritingThreadNum--;

                   return FALSE;

         }

}

 

BOOL CTreadSafeMsgQueue::Reset()

{

         if (!m_bInitedOK) return FALSE;

 

         // 防止重入Reset

         while (m_bStop) Sleep(SLEEP_TIME);

 

         // 防止新的线程进入读写操作状态

         m_bStop = TRUE;

         TRACE("--------------Begin to Reset-------------\n");

 

         // 等待至少一类线程(读线程或写线程)退出同步状态

    while ( (m_WritingThreadNum != 0) && (m_ReadingThreadNum != 0)) Sleep(SLEEP_TIME);

 

         // 此时,可能(最坏可能)还有一类线程处于同步状态,并被阻塞。

         // 它们必定都被阻塞于信号量(Semaphore)状态, 注意此时Event事件处于信号态!!

        

         // 如果被阻塞的是写线程,它们将被释放,并写入数据,但它们写入的数据将被抛弃。

         // 释放写线程将改写m_S_Consumer,所以必须先释放写线程;

         // ReleaseSemaphore(m_S_Producer)m_S_Producer=MAX_QUE_SIZE;

         while (m_WritingThreadNum > 0)

         {

                   // 必须Sleep以出让CPU控制权

                  ReleaseSemaphore(m_S_Producer, 1, NULL);

                  Sleep(5);

         }

 

         // 继续阻塞读线程,令m_S_Consumer=0;

         while (WaitForSingleObject(m_S_Consumer, 10) == WAIT_OBJECT_0);

 

         // 由于写操作会改变m_HeaderToWrite,必须在写操作之后对其重置

         m_HeaderToWrite = m_TailToRead = 0;

        

         // 注意

         TRACE("--------------Finished Reseting-------------\n");

         m_bStop =  FALSE;

 

         return TRUE;

}




相关文章

相关软件