一、基本概念
1、模块:一段可执行的程序(包括EXE和DLL),其程序代码、数据、资源被加载到内存中,由系统建置一个数据结构来管理它,就是一个模块。这里所说的数据结构,名为Module Database(MDB),其实就是PE格式中的PE表头,可以从WINNT.H 档中找到一个IMAGE_NT_HEADER 结构,就是它。
2、进程:就是一大堆拥有权(ownership)的集合。进程拥有地址空间(由memory context决定)、动态配置而来的内存、文件、执行线程、一系列的模块。操作系统使用一个所谓的Process Database(PDB)数据结构,来记录(管理)它所拥有的一切。
3、线程:系统以一个特定的数据结构(Thread Database,TDB)记录执行线程的所有相关资料,包括执行线程局部储存空间(Thread Local Storage,TLS)、消
息队列、handle表格、地址空间(Memory Context )等。
进程主要表达「拥有权」的观念,执行线程则主要表达模块中的程序代码的「执行事实」。
4、CPU调度单位是执行线程。调度器据以排序的,是每个执行线程的优先权。
5、一般,timeslice是20个milliseconds。
6、PDB连接模块示例
7、当Windows 加载器将程序加载内存中,KERNEL32挖出一些内存,构造出一个PDB、一个TDB、一个以上的MDBs(视此程序使用到多少DLL而定)。针对TDB,操作系统又要产生出memory context(就是在操作系统书籍中提到的那些所谓page tables)、消息队列、handle表格、环境数据结(EDB)。当这些系统内部数据结构都构造完毕,指令指位器(Instruction Pointer)移到程序的进入点,才开始程序的执行。
8、会被冻结,表示这个执行线程「要去抓取消息,而执行线程所附带的消息队列中却没有消息」。
冻结有两种方法:一种是SuspendThread,另外一种是暂时冻结Sleep。
9、线程上下文:狭义来讲是指一组缓存器值(包括指令指位器IP)。
10、Worker Threads和UI Threads
从Windows 操作系统的角度来看,执行线程并未分类。但从MFC的角度看,则把执行线程划分为和使用者接口无关的worker threads,以及和使用者接口
(UI)有关的UI threads。
基本上,当我们以::CreateThread 产生一个执行线程,并指定一个执行线程函数,它就是一个worker thread,除非在它的生命中接触到了输入消息-这时候它应该有一个消息循环,以抓取消息,于是该执行线程摇身一变而为UI thread。
注意,执行线程本来就带有消息队列。而如果执行线程程序代码中带有一个消息循环,就称为UI thread。
11、用多线程的时机
把所有UI(User Interface)动作都集中在主执行线程中,其它的「纯种运算工作」考虑交给worker threads。
12、创建多线程
应该先产生一个CWinThread对象,再调用其成员函数CreateThread或全域函数AfxBeginThread将执行线程产生出来。因为CWinThread::CreateThread 和AfxBeginThread不只是::CreateThread 的一层包装,更做了一些application framework所需的内部数据初始化工作,并确保正确的C runtime library版本。
1)建立Worker Threads
利用函数CreateThread或全域函数AfxBeginThread去做。
示例程序
CWinThread* pThread = AfxBeginThread(ThreadFunc, &Param); ... UINT ThreadFunc (LPVOID pParam) { ... }
2)建立UI Threads
产生CWinThread 对象,为了借助其中的消息循环,CWinThread::Run。
示例程序
class CMyThread : public CWinThread { DECLARE_DYNCREATE(CMyThread) public: void BOOL InitInstance(); }; IMPLEMENT_DYNCREATE(CMyThread, CWinThread) BOOL CMyThread::InitInstance() { ... } CWinThread *pThread = AfxBeginThread(RUNTIME_CLASS(CMyThread));
3)线程的结束
对于worker thread,执行线程函数return,执行线程也就结束了。或者执行线程函数也可以调用AfxEndThread,结束一个执行线程。
UI执行线程因为有消息循环的关系,必须在消息队列中放一个WM_QUIT,才能结束执行线程。放置的方式和一般Win32程序一样,调用::PostQuitMessage 即可办到。亦或者,在执行线程的任何一个函数中调用AfxEndThread,也可以结束执行线程。
不论worker thread 或UI thread,都需要一个CWinThread 对象,当执行线程结束,记得把该对象释放掉(利用delete)。
参考
[1] 深入浅出MFC