框范围拖动到合适的大小时松开鼠标左键。
四、在视频Lesson16的事件代码中,有一个问题,修改如下:
void main()
{
HANDLE hThread1;
HANDLE hThread2;
g_hEvent=CreateEvent(NULL,FALSE,FALSE,NULL); //将CreateEvent()函数放置在这个位置
hThread1=CreateThread(NULL,0,Fun1Proc,NULL,0,NULL);
hThread2=CreateThread(NULL,0,Fun2Proc,NULL,0,NULL);
CloseHandle(hThread1);
CloseHandle(hThread2);
//g_hEvent=CreateEvent(NULL,FALSE,FALSE,NULL);//取消这个位置的CreateEvent()函数。
/*g_hEvent=CreateEvent(NULL,FALSE,FALSE,"tickets");
if(g_hEvent)
{
if(ERROR_ALREADY_EXISTS==GetLastError())
{
cout<<"only instance can run!"< return;
}
}*/
SetEvent(g_hEvent);
Sleep(4000);
CloseHandle(g_hEvent);
}
原因:如果在线程产生之后调用CreateEvent(),假如线程提前被操作
系统调度,那么线程里的 WaitForSingleObject 等待的将是一个空的g_hEvent, 在这种情况下,WaitForSingleObject 将返回WAIT_FAILED 。这个问题可以通过在CreateEvent()函数前添加Sleep(10)的调用来查看。
五、在视频Lesson16的采用事件的多线程同步代码中,有一个问题:在线程1和线程2的代码中,有下面一段:
while(TRUE)
{
WaitForSingleObject(g_hEvent,INFINITE);
if(tickets>0)
{
Sleep(1);
cout<<"thread1 sell ticket : "< }
else
break;
SetEvent(g_hEvent);
}
这个代码有一个问题,当线程1或线程2卖完最后一张票时,调用SetEvent(g_hEvent);将事件对象设置为有信号状态,另一个线程等待到事件对象,开始执行代码,判断tickets不大于0,于是执行else语句下的break,退出循环,此时SetEvent(g_hEvent);就没有被执行,导致线程1或线程2一直等待,直到主线程终止运行,整个程序才退出。应将SetEvent(g_hEvent);在 if 和 else 中分别调用,修改如下:
DWORD WINAPI Fun1Proc(
LPVOID lpParameter // thread data
)
{
while(TRUE)
{
WaitForSingleObject(g_hEvent,INFINITE);
// ResetEvent(g_hEvent);
if(tickets>0)
{
Sleep(1);
cout<<"thread1 sell ticket : "< SetEvent(g_hEvent);
}
else
{
SetEvent(g_hEvent