需求:简单的扫描功能,一个扫描配置对话框(包含 扫描路径文本框、扫描结果显示LIST控件、扫描启动按钮)
在扫描启动按钮的单击响应函数中,通过如下方式启用多线程:
CThreadX **mythread=(CThreadX **)malloc(threadnum*4);//申请多线程指针数组
HANDLE *myhth=(HANDLE*)malloc(threadnum*sizeof(HANDLE)); //申请HANDLE类型数组
unsigned *uithreadID=(unsigned *)malloc(threadnum*sizeof(unsigned));//申请无符号整形数组
for(int i=0;i<threadnum;i++)//创建多线程
{

mythread[i]=new CThreadX(threadnum,i ,Param.dwStartPort,Param.dwEndPort,Param.dwStartIp,Param.dwEndIp,hRecvStopEvent ,this->m_hWnd); 
myhth[i]=(HANDLE)_beginthreadex( NULL, // security  
0, // stack size  
CThreadX::ThreadStaticEntryPoint,  
mythread[i], // arg list  
CREATE_SUSPENDED, // so we can later call ResumeThread()  
&uithreadID[i] );
}
........
CThreadX是自定义的一个类,实现代码如下:
unsigned __stdcall CThreadX::ThreadStaticEntryPoint(void * pThis)  
{  
CThreadX * pthX = (CThreadX*)pThis; // 指针类型转换  
pthX->Thread_ComputeScanIP();   // 调用真正的功能函数
return 1; 
}
void CThreadX::Thread_ComputeScanIP()
{/*
int sub=(th_ipend-th_ipstart)/th_count;
DWORD ipend=th_ipstart+sub*(th_seq+1)-1;
DWORD ipstart=th_ipstart+sub*th_seq;
Thread_SendPacket(ipstart,ipend,th_portstart,th_portend);*/
CListCtrl * u_LIST_IPDetectResult;
u_LIST_IPDetectResult=(CListCtrl *)::GetDlgItem(th_handle,IDC_LIST_IPDetectResult);
u_LIST_IPDetectResult->InsertItem(count++,"test");//这里运行时会出现严重错误
}
请问原因

解决方案 »

  1.   


    1) 不要在线程函数体内操作MFC控件,不要再线程里面调用UpdateData函数更新用户界面,而应该尽量采用发送消息的方式,在主线程的消息响应函数中操作控件;2)不建议采用SendMessage往主线程发送消息,因为它是同步的,阻塞的,可以考虑采用PostMessage代替;3)线程退出时,尽量不要使用TerminateThread函数,而尽可能的让线程自己退出;4) 当线程退出时,必须先等待工作者线程退出,主线程才退出,但是在主线程里面不要使用WaitForSingleObject或WaitForMultiObjects等待线程结束,因为它可能造成死锁,当主线程使用这两个函数时,主线程就挂起了,尤其在第 (1), (2) 种情况下,工作者线程还在调用主线程里面的资源,这样造成死锁;5) 为了防止退出死锁的发生,尽量使用MsgWaitForMultipleObjects函数,因为该函数等待时,可以等待线程句柄 有信号,而且还可以等待消息,不会造成死锁;