一般而言,应用程序中的一个次要线程总是为主线程执行特定的任务,这样,主线程和次要线程间必定有一个信息传递的渠道,也就是主线程和次要线程间要进行通信。这种线程间的通信不但是难以避免的,而且在多线程编程中也是复杂和频繁的,下面将进行说明。
- 使用全局变量进行通信
由于属于同一个进程的各个线程共享操作系统分配该进程的资源,故解决线程间通信最简单的一种方法是使用全局变量。对于标准类型的全局变量,我们建议使用volatile 修饰符,它告诉编译器无需对该变量作任何的优化,即无需将它放到一个寄存器中,并且该值可被外部改变。如果线程间所需传递的信息较复杂,我们可以定义一个结构,通过传递指向该结构的指针进行传递信息。
- 使用自定义消息
我们可以在一个线程的执行函数中向另一个线程发送自定义的消息来达到通信的目的。一个线程向另外一个线程发送消息是通过操作系统实现的。利用Windows操作系统的消息驱动机制,当一个线程发出一条消息时,操作系统首先接收到该消息,然后把该消息转发给目标线程,接收消息的线程必须已经建立了消息循环。
例程7 MultiThread7 .基于MultiThread7Dlg对话框的应用程序
该例程演示了如何使用自定义消息进行线程间通信。首先,主线程向CCalculateThread线程发送消息WM_CALCULATE,CCalculateThread线程收到消息后进行计算,再向主线程发送WM_DISPLAY消息,主线程收到该消息后显示计算结果。
1. 线程类.h:
#define WM_CALCULATE WM_USER+1
class CCalculateThread : public CWinThread
{
DECLARE_DYNCREATE(CCalculateThread)
protected:
CCalculateThread(); // protected constructor used by dynamic creation
public:
public:
virtual BOOL InitInstance();
virtual int ExitInstance();
protected:
afx_msg void OnCalculate(WPARAM wParam,LPARAM lParam);
virtual ~CCalculateThread();
DECLARE_MESSAGE_MAP()
};
2. 线程类.cpp:
#include "CalculateThread.h"
#include "MultiThread7Dlg.h"
IMPLEMENT_DYNCREATE(CCalculateThread, CWinThread)
CCalculateThread::CCalculateThread()
{
}
CCalculateThread::~CCalculateThread()
{
}
BOOL CCalculateThread::InitInstance()
{
// TODO: perform and per-thread initialization here
return TRUE;
}
int CCalculateThread::ExitInstance()
{
// TODO: perform any per-thread cleanup here
return CWinThread::ExitInstance();
}
BEGIN_MESSAGE_MAP(CCalculateThread, CWinThread)
ON_THREAD_MESSAGE(WM_CALCULATE, OnCalculate)
END_MESSAGE_MAP()
// CCalculateThread message handlers
void CCalculateThread::OnCalculate(WPARAM wParam,LPARAM lParam)
{
int nTmpt=0;
for(int i=0;i<=(int)wParam;i++)
{
nTmpt=nTmpt+i;
}
Sleep(500);
::PostMessage((HWND)(GetMainWnd()->GetSafeHwnd()), WM_DISPLAY, nTmpt, NULL);
}
1. 在MultiThread7Dlg.h中添加:
#include "CalculateThread.h"
#define WM_DISPLAY WM_USER+2
// CMultiThread7Dlg dialog
class CMultiThread7Dlg : public CDialog
{
CCalculateThread* m_pCalculateThread;
protected:
int nAddend /* =50*/;
LRESULT OnDisplay(WPARAM wParam,LPARAM lParam);
afx_msg void OnSum();
};
2. 在MultiThread7Dlg.h中添加:
#include "stdafx.h"
#include "MultiThread7.h"
#include "MultiThread7Dlg.h"
// CMultiThread7Dlg dialog
BEGIN_MESSAGE_MAP(CMultiThread7Dlg, CDialog)
ON_BN_CLICKED(IDC_SUM, OnSum)
ON_MESSAGE(WM_DISPLAY, OnDisplay)
END_MESSAGE_MAP()
LRESULT CMultiThread7Dlg::OnDisplay(WPARAM wParam,LPARAM lParam)
{
int nTemp=(int)wParam;
SetDlgItemInt(IDC_STATUS,nTemp,FALSE);
return 0;
}
void CMultiThread7Dlg::OnSum() /*button*/
{
m_pCalculateThread=
(CCalculateThread*)AfxBeginThread(RUNTIME_CLASS(CCalculateThread));
Sleep(500);
m_pCalculateThread->PostThreadMessage(WM_CALCULATE,nAddend,NULL);
}