多线程死锁应该注意的问题和解决方法
引言
在多线程程序中,要格外注意执行的主体,一不小心就会出现线程死锁或者卡死的情况。如在主线程中通过
hThread = (HANDLE)_beginthread(dosomething, 0, this);
dwThreadId = GetThreadId(hThread);
启动子线程,子线程dosomething中如果等待自己则会出现线程陷入死锁的问题。
if (hThread)
WaitForSingleObject(hThread, INFINITE);
那么该如何避免和解决这种跨线程导致的问题呢?
多线程安全的解决方法
但有些函数确实需要跨线程执行,为了线程安全,这些函数就需要知道调用者是否是本线程在调用。如果是本线程在调用,则无需考虑线程执行状态,因为线程数据对自己是绝对安全的。
那么如何知道当前方法是否运行在线程中呢?简单的方法就是查看当前线程是否就是与要等待的线程句柄相同,如果是相同的,则认为本方法运行在本线程,则方法运行绝对安全。无需做出处理,直接略过就可以。
一定要注意不要通过线程句柄去判断是否相同线程,要通过线程ID来判断。
DWORD dwCurId = GetCurrentThreadId();
if (dwCurId == dwThreadId)
return;//can not kill self
消息机制
为了保证跨线程方法调用安全性,线程间方法调用建议采用消息机制来实现,在windows中可以通过windows消息机制来实现。
但是一定要注意:
如果需要在子线程中向主线程发消息,一定要通过PostMessage发消息,不要通过SendMessage(不跨进程)失去消息跨线程处理的优势。
- 发送消息通过PostMessage来实现:
PostMessage(_hwnd, WM_COMMAND, nCmdID, dwParam);
hWnd:目标窗口
Msg:消息类型,如WM_COMMAND命令消息
wParam:可自定义命令号nCmdID
lParam:可传递自定义参数
BOOL WINAPI PostMessage(
_In_opt_ HWND hWnd,
_In_ UINT Msg,
_In_ WPARAM wParam,
_In_ LPARAM lParam);
- 消息接收可在MFC中通过窗口处理函数虚方法来方便的实现:
BOOL CPdfView::OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult)
{
// TODO: 在此添加专用代码和/或调用基类
if (WM_COMMAND == message && wParam == ID_HIGHLIGHT)
{
CPdfDoc* pDoc = (CPdfDoc*)GetDocument();
if (pDoc != NULL && pDoc->m_p != NULL)
{
DocNode* node = (DocNode*)lParam;
pDoc->m_p->highlight(node);
}
return TRUE;
}
return CView::OnWndMsg(message, wParam, lParam, pResult);
}
message得到当前消息,判断是否命令
wParam 得到当前命令ID
wParam 得到当前参数:
DocNode* node = (DocNode*)msg->lParam;
通过消息机制有效处理主线程和子线程的执行序列,避免主线程和子线程混合执行导致出现死锁和死循环。