中航鑫源建设集团有限公司网站,舆情监测软件价格,万网提供的网站建设服务的具体项目,郑州做网站九零后本文将带领你与多线程作第一次亲密接触#xff0c;并深入分析CreateThread与_beginthreadex的本质区别#xff0c;相信阅读本文后你能轻松的使用多线程并能流畅准确的回答CreateThread与_beginthreadex到底有什么区别#xff0c;在实际的编程中到底应该使用CreateThread还是… 本文将带领你与多线程作第一次亲密接触并深入分析CreateThread与_beginthreadex的本质区别相信阅读本文后你能轻松的使用多线程并能流畅准确的回答CreateThread与_beginthreadex到底有什么区别在实际的编程中到底应该使用CreateThread还是_beginthreadex 使用多线程其实是非常容易的下面这个程序的主线程会创建了一个子线程并等待其运行完毕子线程就输出它的线程ID号然后输出一句经典名言——Hello World。整个程序的代码非常简短只有区区几行。 [cpp] view plaincopy //最简单的创建多线程实例 #include stdio.h #include windows.h //子线程函数 DWORD WINAPI ThreadFun(LPVOID pM) { printf(子线程的线程ID号为%d\n子线程输出Hello World\n, GetCurrentThreadId()); return 0; } //主函数所谓主函数其实就是主线程执行的函数。 int main() { printf( 最简单的创建多线程实例\n); printf( -- by MoreWindows( http://blog.csdn.net/MoreWindows ) --\n\n); HANDLE handle CreateThread(NULL, 0, ThreadFun, NULL, 0, NULL); WaitForSingleObject(handle, INFINITE); return 0; } 运行结果如下所示 下面来细讲下代码中的一些函数
第一个 CreateThread
函数功能创建线程
函数原型
HANDLEWINAPICreateThread( LPSECURITY_ATTRIBUTESlpThreadAttributes, SIZE_TdwStackSize, LPTHREAD_START_ROUTINElpStartAddress, LPVOIDlpParameter, DWORDdwCreationFlags, LPDWORDlpThreadId
);
函数说明
第一个参数表示线程内核对象的安全属性一般传入NULL表示使用默认设置。
第二个参数表示线程栈空间大小。传入0表示使用默认大小1MB。
第三个参数表示新线程所执行的线程函数地址多个线程可以使用同一个函数地址。
第四个参数是传给线程函数的参数。
第五个参数指定额外的标志来控制线程的创建为0表示线程创建之后立即就可以进行调度如果为CREATE_SUSPENDED则表示线程创建后暂停运行这样它就无法调度直到调用ResumeThread()。
第六个参数将返回线程的ID号传入NULL表示不需要返回该线程ID号。
函数返回值
成功返回新线程的句柄失败返回NULL。 第二个 WaitForSingleObject
函数功能等待函数 –使线程进入等待状态直到指定的内核对象被触发。
函数原形
DWORDWINAPIWaitForSingleObject( HANDLEhHandle, DWORDdwMilliseconds
);
函数说明
第一个参数为要等待的内核对象。
第二个参数为最长等待的时间以毫秒为单位如传入5000就表示5秒传入0就立即返回传入INFINITE表示无限等待。
因为线程的句柄在线程运行时是未触发的线程结束运行句柄处于触发状态。所以可以用WaitForSingleObject()来等待一个线程结束运行。
函数返回值
在指定的时间内对象被触发函数返回WAIT_OBJECT_0。超过最长等待时间对象仍未被触发返回WAIT_TIMEOUT。传入参数有错误将返回WAIT_FAILED CreateThread()函数是Windows提供的API接口在C/C语言另有一个创建线程的函数_beginthreadex()在很多书上包括《Windows核心编程》提到过尽量使用_beginthreadex()来代替使用CreateThread()这是为什么了下面就来探索与发现它们的区别吧。 首先要从标准C运行库与多线程的矛盾说起标准C运行库在1970年被实现了由于当时没任何一个操作系统提供对多线程的支持。因此编写标准C运行库的程序员根本没考虑多线程程序使用标准C运行库的情况。比如标准C运行库的全局变量errno。很多运行库中的函数在出错时会将错误代号赋值给这个全局变量这样可以方便调试。但如果有这样的一个代码片段 [cpp] view plaincopy if (system(notepad.exe readme.txt) -1) { switch(errno) { ...//错误处理代码 } } 假设某个线程A在执行上面的代码该线程在调用system()之后且尚未调用switch()语句时另外一个线程B启动了这个线程B也调用了标准C运行库的函数不幸的是这个函数执行出错了并将错误代号写入全局变量errno中。这样线程A一旦开始执行switch()语句时它将访问一个被B线程改动了的errno。这种情况必须要加以避免因为不单单是这一个变量会出问题其它像strerror()、strtok()、tmpnam()、gmtime()、asctime()等函数也会遇到这种由多个线程访问修改导致的数据覆盖问题。 为了解决这个问题Windows操作系统提供了这样的一种解决方案——每个线程都将拥有自己专用的一块内存区域来供标准C运行库中所有有需要的函数使用。而且这块内存区域的创建就是由C/C运行库函数_beginthreadex()来负责的。下面列出_beginthreadex()函数的源代码我在这份代码中增加了一些注释以便读者更好的理解_beginthreadex()函数与CreateThread()函数的区别。 [cpp] view plaincopy //_beginthreadex源码整理By MoreWindows( http://blog.csdn.net/MoreWindows ) _MCRTIMP uintptr_t __cdecl _beginthreadex( void *security, unsigned stacksize, unsigned (__CLR_OR_STD_CALL * initialcode) (void *), void * argument, unsigned createflag, unsigned *thrdaddr ) { _ptiddata ptd; //pointer to per-thread data 见注1 uintptr_t thdl; //thread handle 线程句柄 unsigned long err 0L; //Return from GetLastError() unsigned dummyid; //dummy returned thread ID 线程ID号 // validation section 检查initialcode是否为NULL _VALIDATE_RETURN(initialcode ! NULL, EINVAL, 0); //Initialize FlsGetValue function pointer __set_flsgetvalue(); //Allocate and initialize a per-thread data structure for the to-be-created thread. //相当于new一个_tiddata结构并赋给_ptiddata指针。 if ( (ptd (_ptiddata)_calloc_crt(1, sizeof(struct _tiddata))) NULL ) goto error_return; // Initialize the per-thread data //初始化线程的_tiddata块即CRT数据区域 见注2 _initptd(ptd, _getptd()-ptlocinfo); //设置_tiddata结构中的其它数据这样这块_tiddata块就与线程联系在一起了。 ptd-_initaddr (void *) initialcode; //线程函数地址 ptd-_initarg argument; //传入的线程参数 ptd-_thandle (uintptr_t)(-1); #if defined (_M_CEE) || defined (MRTDLL) if(!_getdomain((ptd-__initDomain))) //见注3 { goto error_return; } #endif // defined (_M_CEE) || defined (MRTDLL) // Make sure non-NULL thrdaddr is passed to CreateThread if ( thrdaddr NULL )//判断是否需要返回线程ID号 thrdaddr dummyid; // Create the new thread using the parameters supplied by the caller. //_beginthreadex()最终还是会调用CreateThread()来向系统申请创建线程 if ( (thdl (uintptr_t)CreateThread( (LPSECURITY_ATTRIBUTES)security, stacksize, _threadstartex, (LPVOID)ptd, createflag, (LPDWORD)thrdaddr)) (uintptr_t)0 ) { err GetLastError(); goto error_return; } //Good return return(thdl); //线程创建成功,返回新线程的句柄. //Error return error_return: //Either ptd is NULL, or it points to the no-longer-necessary block //calloc-ed for the _tiddata struct which should now be freed up. //回收由_calloc_crt()申请的_tiddata块 _free_crt(ptd); // Map the error, if necessary. // Note: this routine returns 0 for failure, just like the Win32 // API CreateThread, but _beginthread() returns -1 for failure. //校正错误代号(可以调用GetLastError()得到错误代号) if ( err ! 0L ) _dosmaperr(err); return( (uintptr_t)0 ); //返回值为NULL的效句柄 } 讲解下部分代码
注1_ptiddataptd;中的_ptiddata是个结构体指针。在mtdll.h文件被定义 typedefstruct_tiddata *_ptiddata
微软对它的注释为Structure for each threads data。这是一个非常大的结构体有很多成员。本文由于篇幅所限就不列出来了。 注2_initptd(ptd,_getptd()-ptlocinfo);微软对这一句代码中的getptd()的说明为 /* return address of per-thread CRT data */ _ptiddata __cdecl_getptd(void);
对_initptd()说明如下 /* initialize a per-thread CRT data block */ void__cdecl_initptd(_Inout_ _ptiddata_Ptd,_In_opt_ pthreadlocinfo_Locale);
注释中的CRTC Runtime Library即标准C运行库。 注3if(!_getdomain((ptd-__initDomain)))中的_getdomain()函数代码可以在thread.c文件中找到其主要功能是初始化COM环境。 由上面的源代码可知_beginthreadex()函数在创建新线程时会分配并初始化一个_tiddata块。这个_tiddata块自然是用来存放一些需要线程独享的数据。事实上新线程运行时会首先将_tiddata块与自己进一步关联起来。然后新线程调用标准C运行库函数如strtok()时就会先取得_tiddata块的地址再将需要保护的数据存入_tiddata块中。这样每个线程就只会访问和修改自己的数据而不会去篡改其它线程的数据了。因此如果在代码中有使用标准C运行库中的函数时尽量使用_beginthreadex()来代替CreateThread()。相信阅读到这里时你会对这句简短的话有个非常深刻的印象如果有面试官问起你也可以流畅准确的回答了^_^。 接下来类似于上面的程序用CreateThread()创建输出“Hello World”的子线程下面使用_beginthreadex()来创建多个子线程 [cpp] view plaincopy //创建多子个线程实例 #include stdio.h #include process.h #include windows.h //子线程函数 unsigned int __stdcall ThreadFun(PVOID pM) { printf(线程ID号为%4d的子线程说Hello World\n, GetCurrentThreadId()); return 0; } //主函数所谓主函数其实就是主线程执行的函数。 int main() { printf( 创建多个子线程实例 \n); printf( -- by MoreWindows( http://blog.csdn.net/MoreWindows ) --\n\n); const int THREAD_NUM 5; HANDLE handle[THREAD_NUM]; for (int i 0; i THREAD_NUM; i) handle[i] (HANDLE)_beginthreadex(NULL, 0, ThreadFun, NULL, 0, NULL); WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE); return 0; } 运行结果如下 图中每个子线程说的都是同一句话不太好看。能不能来一个线程报数功能即第一个子线程输出1第二个子线程输出2第三个子线程输出3……。要实现这个功能似乎非常简单——每个子线程对一个全局变量进行递增并输出就可以了。代码如下 [cpp] view plaincopy //子线程报数 #include stdio.h #include process.h #include windows.h int g_nCount; //子线程函数 unsigned int __stdcall ThreadFun(PVOID pM) { g_nCount; printf(线程ID号为%4d的子线程报数%d\n, GetCurrentThreadId(), g_nCount); return 0; } //主函数所谓主函数其实就是主线程执行的函数。 int main() { printf( 子线程报数 \n); printf( -- by MoreWindows( http://blog.csdn.net/MoreWindows ) --\n\n); const int THREAD_NUM 10; HANDLE handle[THREAD_NUM]; g_nCount 0; for (int i 0; i THREAD_NUM; i) handle[i] (HANDLE)_beginthreadex(NULL, 0, ThreadFun, NULL, 0, NULL); WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE); return 0; } 对一次运行结果截图如下 显示结果从1数到10看起来好象没有问题。 答案是不对的虽然这种做法在逻辑上是正确的但在多线程环境下这样做是会产生严重的问题下一篇《秒杀多线程第三篇 原子操作 Interlocked系列函数》将为你演示错误的结果可能非常出人意料并解释产生这个结果的详细原因。