深度解析MFC线程及机制

发表于:2007-07-01来源:作者:点击数: 标签:
在VC中,启动一线程有3种方式: 1.使用MFC中启动一个线程一般使用AfxBeginThread函数; 2.使用API则使用CreateThread; 3.就是使用_beginthread和_beginthreadex两个函数。 使用MFC的AfxBeginThread函数启动一个线程,不管你代入的参数是否是个从CWinThread继承

在VC中,启动一线程有3种方式:

1.使用MFC中启动一个线程一般使用AfxBeginThread函数;

2.使用API则使用CreateThread;

3.就是使用_beginthread和_beginthreadex两个函数。

使用MFC的AfxBeginThread函数启动一个线程,不管你代入的参数是否是个从CWinThread继承来的线程类还是一以标准的线程函数地址及参数(UINT ThreadProc(LPVOID pParam)的形式),都返回了一CWinThread指针方便了我们使用。但其中的机理没有在哪本书上有人仔细分析过,没办法只好查看MFC源码了。经过分析,现将分析的结果与大家分享。:)

在CWinThread类中有两个构造函数:

一为默认的构造,还有一个为一标准线程函数地址和输入参数的构造函数-
CWinThread(AFX_THREADPROC pfnThreadProc, LPVOID pParam)。当我们使用AfxBeginThread启动一个线程时,无外2种方法:
 1.直接启动已CWinThread为基类的线程类-AfxBeginThread(RUNTIME_CLASS(...),...);
 2.直接启动标准的线程函数-AfxBeginThread((AFX_THREADPROC),LPVOID ,....);

还有一种方法可能大家不常用,就是以CWinThread继承一线程类,在内存中new,然后调用CWinThread的成员函数CreateThread来启动自身;
经过我的分析,实际上第1种和第2种方法就是第3中方法的详细化,调用更加灵活,但1、2种比起来要方便些。:)
实际上AfxBeginThread根据参数实现创建一CWinThread线程:若参数为一标准的线程函数地址,则new一CWinThread对象,以函数地址及参数构造;若参数为一CWinThread继承过来的类则直接调用CRunTimeClass的CreateObject创建一CWinThread对象,以默认参数构造。然后调用CreateThread成员函数,调用不成功便调用Delete(),判断是否设置了自我删除对象(m_bAutoDelete = TRUE),为真则删除自身。

在CreateThread函数中,首先创建2个事件,MFC利用_beginthreadex启动一名为_AfxThreadEntry的全局线程函数,并通过一_AFX_THREAD_STARTUP结构体将一系列参数传入,然后等待_AfxThreadEntry将一线程初始化完毕设置事件1状态为止,然后自己设置另一个事件2状态,触发_AfxThreadEntry中的代码;在_AfxThreadEntry中,函数首先通过一系列函数初始化线程,再设置事件1状态后等待CWinThread的CreateThread执行完毕后的事件2,这时判断_AFX_THREAD_STARTUP中的线程函数地址是否为空,若不为空就直接执行函数代码;若不为空就执行CWinThread的InitInstance判断是否返回为TRUE,是的话就进入消息循环知道发送给线程WM_QUIT消息为止;若返回为FALSE,就调用ExitInstance;最后都调用AfxEndThread结束线程。而AfxEndThread则从模块中先取得CWinThread*类型的指针判断是否为NULL,否的话则调用CWinThread的Delete来将CWinThread从内存中删除。

以下是简易流程图:

AfxBeginThread:

_AfxThreadEntry:

希望对大家有用!


原文转自:http://www.ltesting.net