Windows下创建线程的方法_beginThread和CreateThread

windows下创建线程有CreateThread  _beginthreadex两个函数,起始_beginthreadex函数是CreateThread的封装,使用createthread容易造成内存泄露,一般不推荐使用,推荐使用_beginthreadex函数来创建线程。而mfc下推荐使用AfxBeginThread()。

内存泄露其实是冲突的问题。

在很多参考书上,都说不要用CreateThread 创建线程、并用CloseHandle来关闭这个线程,因为这样做会导致内存泄漏,而应该用_beginthread来创建线程,_endthread来销毁线程。其实,真正的原因并非如此。看如下一段代码:
HANDLE CreateThread(
LPSECURITY_ATTRIBUTES lpThreadAttributes, //  线程安全 属性
DWORD dwStackSize, // 堆栈大小
LPTHREAD_START_ROUTINE lpStartAddress, // 线程函数
LPVOID lpParameter, //线程参数
DWORD dwCreationFlags, // 线程创建属性        (1)CREATE_SUSPENDED(0x00000004):创建一个挂起线程     (2)0:表示创建后立即激活
LPDWORD lpThreadId // 线程ID                                                                                                     
      线程 中止运行后,线程对象仍然在系统中,必须通过CloseHandle函数来关闭该线程对象。CloseHandle函数的原型是:
BOOL CloseHandle( HANDLE hObject );//HANDLE hObject 对象句柄
CloseHandle可以关闭多种类型的对象,比如文件对象等,这里使用这个函数来关闭线程对象。调用时,hObject为待关闭的线程对象的句柄。
说使用这种方法可能会引发内存泄漏问题,其实不完全正确。那为什么会引起内存的泄漏呢?因为当线程的函数用到了C的标准库的时候,很容易导致冲突,所以在创建VC的工程时,系统提示是用 单线程 还是用多线程的库,因为在C的内部有很多的 全局变量 。例如,出错号、 文件句柄 全局变量
因为在C的库中有全局变量,这样用C的库时,如果程序中使用了标准的C程序库时,就很容易导致运行不正常,会引起很多的冲突。所以, 微软 和Borland都对C的库进行了一些改进。但是这个改进的一个条件就是,如果一个线程已经开始创建了,就应该创建一个结构来包含这些 全局变量 ,接着把这些全局变量放入线程的上下文中和这个线程相关起来。这样, 全局变量 就会依赖于这个线程,不会引起冲突。
这样做就会有一个问题,什么时候这个线程开始创建呢?标准的Windows的API是不知道的,因为它是静态的库。这些库都是放在VC的LIB的目录内的,而线程函数是 操作系统 的函数。所以,VC和BC在创建线程时,都会用_beginThread来创建线程,再用_endThread来结束线程。这样,它们在创建线程的时候,就会知道什么时候创建了线程,并把 全局变量 放入某一结构中,让它和线程能关联起来。这样就不会发生冲突了。
很显然,要完成这个功能,首先需要分配结构表把 全局变量 包含起来。这个过程是在_beginThread时做的,而释放在_endTread内完成。
所以,当用_beginThread来创建,而用CloseHandle来关闭线程时,这时复制的全局结构就不会被释放了,这就有了内存的泄漏。这就是很多资料所说的内存泄漏问题的真正的原因。
其实,可以不用_beginThread和_endThread这一对函数。如果用CreateThread函数创建,用CloseHandle关闭,那么,与C有关的库就会用全局的,它们会引起冲突。所以,比较好的方法就是在 线程 内不用标准的C的库(可以使用Windows API的 库函数 )。这样就不会有什么问题,也就不会引起冲突。例如,字符串的操作函数、文件操作等。

当某个程序创建一个线程后,会产生一个线程的句柄,线程的句柄主要用来控制整个线程的运行,例如停止、挂起或设置线程的优先级等操作。


转载地址:http://blog.csdn.net/hanyujianke/article/details/8502189

你可能感兴趣的:(Windows下创建线程的方法_beginThread和CreateThread)