`

多线程 CreateThread与_beginthreadex本质区别

 
阅读更多

本文参考了 http://blog.csdn.net/morewindows/article/details/7421759

主函数创建一个线程,并且等待它执行完毕

 

#include <iostream>
#include <windows.h>

using namespace std;

DWORD WINAPI threadFunc(PVOID pvParam) {
	cout << "hello " << (*((char *) pvParam))++ << "  " << GetCurrentThreadId() << endl;
	return 0;
}

int main() {

	char b[] = "abc";
	char *c = b;
	HANDLE handle = CreateThread(NULL, 0, threadFunc, (PVOID) &c, 0, NULL);
	WaitForSingleObject(handle, INFINITE);

	cout << c << endl;

	getchar();
	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为例。有的函数会在出错时,设置该变量。现在假定有这样的一个代码段:

bool fFailure = (system("NOTEPAD.EXE README.TXT") == -1);

if(fFailure) {
	switch(errno) {
		。。。
	}
}

 

假设在调用了system函数之后,并在执行if语句之前,执行上述代码的线程被中断了。另外还假设,这个线程被中断后,同一个进程中的另一个线程开始执行,而且这个新线程将执行另一个c运行库函数,后者设置了全局变量errno。当cpu后来被分配回第一个线程时,对于上述代码中的system函数调用,其errno反应的就不再是正确的错误码。

 

为了解决这个问题,每个线程都要有自己的errno变量,同时不能让他去修改另一个线程的errno变量。

这仅仅是证明了“标准c/c++运行库最初不是为多线程应用程序设计”的众多例子中的一个。在多线程环境中会出问题的c/c++运行库变量的函数有errno,_doserrno, strtok, _wcstok. strerror, _strerror, tmpnam, tmpfile, 

asctime, _wasctime, gmtime, _ecvt 和 _fcvt等。

 

为了保证c和c++多线程应用程序正常运行,必须创建一个数据结构,并使之与使用了c/c++运行库函数的每个线程关联。然后,在调用c/c++运行库函数时,那些函数必须知道去查找主调线程的数据块,从而避免影响到其他线程。

 

系统在创建新的线程时,其本身也并不知道要如何去分配这个数据块。系统并不知道应用程序是用c/c++来写的,不知道你调用的函数并非天生就是线程安全的,保证线程安全是程序员的责任。在创建新线程的时候,一定不要调用操作系统的createThread函数。 相反,必须调用c/c++运行库函数_beginthreadex。

 

为了解决这个问题,Windows操作系统提供了这样的一种解决方案——每个线程都将拥有自己专用的一块内存区域来供标准C运行库中所有有需要的函数使用。而且这块内存区域的创建就是由C/C++运行库函数_beginthreadex()来负责的。下面列出_beginthreadex()函数的源代码(我在这份代码中增加了一些注释)以便读者更好的理解_beginthreadex()函数与CreateThread()函数的区别。

 

_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 thread's 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);

注释中的CRT C Runtime Library)即标准C运行库。

 

3if(!_getdomain(&(ptd->__initDomain)))中的_getdomain()函数代码可以在thread.c文件中找到,其主要功能是初始化COM环境。

 

由上面的源代码可知,_beginthreadex()函数最终还是会调用CreateThread()来向系统申请创建线程,_beginthreadex()函数在创建新线程时会分配并初始化一个_tiddata块。这个_tiddata块自然是用来存放一些需要线程独享的数据。事实上新线程运行时会首先将_tiddata块与自己进一步关联起来。然后新线程调用标准C运行库函数如strtok()时就会先取得_tiddata块的地址再将需要保护的数据存入_tiddata块中。这样每个线程就只会访问和修改自己的数据而不会去篡改其它线程的数据了。因此,如果在代码中有使用标准C运行库中的函数时,尽量使用_beginthreadex()来代替CreateThread()相信阅读到这里时,你会对这句简短的话有个非常深刻的印象,如果有面试官问起,你也可以流畅准确的回答了^_^

 

#include <windows.h>
#include <process.h>
#include <iostream>

using namespace std;

unsigned int __stdcall threadFunc(PVOID pM) {
	cout << * ((int *)pM) << endl;
	return 0;
}

int main() {
	int a = 9;

	HANDLE handle = (HANDLE)_beginthreadex(NULL, 0, threadFunc, (PVOID) &a, 0, NULL);
	WaitForSingleObject(handle, INFINITE);

	getchar();
	return 0;
}

 

实例代码如上。本篇完结。

 

分享到:
评论

相关推荐

    CreateThread与_beginthreadex本质区别

    介绍了delphi中CreateThread与_beginthreadex本质区别。

    _beginthreadex与CreateThread区别

    本文将深入探讨两种常见的线程创建函数:`CreateThread`和`_beginthreadex`,以及它们之间的本质区别。 `CreateThread`是Windows API提供的一个函数,允许程序员在进程中创建新的线程。它的函数原型如下: ```cpp ...

    CreateThread函数创建窗口线程.rar_createthread_线程 窗口_线程函数

    在Windows编程中,多线程技术是至关重要的,它允许程序同时执行多个任务,提高程序的效率和响应性。`CreateThread`函数是Windows API提供的一种用于创建新线程的机制,尤其适用于需要同时处理GUI(图形用户界面)和...

    深入分析多线程同步.doc

    本文将深入探讨在C语言环境下,如何使用`CreateThread`和`_beginthreadex`来创建线程,并分析两者之间的本质区别。 `CreateThread`是Windows API提供的一个函数,用于创建一个新的线程。它的主要参数包括线程安全...

    beginthread_和_CreateThread

    标题与描述概述的知识点主要集中在两个关键的线程创建函数——`_beginthread`与`CreateThread`的区别上。这两个函数都是在Windows编程环境中用于创建线程的重要手段,但它们在内部实现机制、资源管理以及与C运行时库...

    快速秒杀windows多线程

    4. CreateThread与_beginthreadex函数的区别:在多线程编程中,CreateThread和_beginthreadex都是用来创建线程的函数,但它们之间存在一些本质的区别。beginthreadex函数是C运行时库提供的函数,而CreateThread是...

    VC++多线程管理器_authornop_Vc_

    1. **线程创建**:在VC++中,可以使用`CreateThread`函数或`_beginthreadex`函数来创建新线程。这两个函数都需要传递一个函数指针,该指针指向新线程要执行的代码入口点。线程创建后,系统会为新线程分配资源,如栈...

    VisualC++多线程编程详解

    #### 一、多线程编程的必要性与问题引入 在软件开发中,特别是在需要处理大量计算或等待I/O操作的场景下,单线程程序往往不能充分利用现代多核处理器的潜力,且容易出现用户界面卡顿的情况。例如,在Visual C++中,...

    多线程下载程序_Internetview_homeiqr_多线程下载_Vc_

    本文将深入探讨标题为"多线程下载程序_Internetview_homeiqr_多线程下载_Vc_"的开源代码,以及与之相关的多线程下载技术。 首先,我们要理解什么是多线程。在计算机科学中,线程是程序执行的最小单元,每个线程可以...

    堪称精品的VB多线程控制台源程序代码.rar_vb 多线程_vb6_vb6多线程_vb多线程_多线程

    例如,可以使用`CreateThread`或`_beginthreadex`函数来创建新的线程。每个线程都有自己的执行上下文,可以在不同的时间片段运行,使得程序能够并行处理任务。 然后,我们需要设计线程间的通信机制。在VB6中,这...

    C++多线程总结[归纳].pdf

    本文档对C++多线程编程进行了总结,介绍了三种创建线程的方法:CreateThread函数、AfxBeginThread函数和_beginthread()/_beginthreadex()函数,同时对线程的管理和终止进行了详细的讲解。 CreateThread函数 ...

    CreateThread3线程DEMO1.rar_createthread_线程 builder

    在Windows API中,`CreateThread`函数是创建新线程的关键接口,用于执行特定的代码段,即线程函数。这个DEMO1示例可能是为了演示如何使用`CreateThread`来构建线程。线程是操作系统调度的基本单位,它允许程序并发...

    _beginthread_和_CreateThread_区别与使用方法

    _beginthread 和 CreateThread 区别与使用方法 _beginthread 和 CreateThread 是 Windows 程序中创建线程的两种方法,但它们之间存在着一些重要的区别。本文将从 CRT 源代码出发,探讨这两个函数的差异,并解释它们...

    Windows多线程编程技术与实例(C++)(PDF)

    接着,书中详细讲解了C++中用于多线程编程的API,如CreateThread、_beginthreadex等,以及如何使用C++11及更新的标准库中的std::thread。这些API的使用方法和注意事项是多线程编程实践中的重要知识,书中会有具体的...

    pb真正的多线程,用createthread创建的多线程.rar

    标题中的“pb真正的多线程,用createthread创建的多线程.rar”指的是PowerBuilder(PB)编程环境中实现的多线程技术,通过使用Windows API函数`CreateThread`来创建新的执行线程。在PowerBuilder中,多线程允许应用...

    RawSocketServerExample.rar_Vc_socket vc_vc 多线程_多线程 Socket_多线程 vi

    在"RawSocketServerExample"中,多线程的实现可能利用了Windows API的CreateThread函数,或者C++标准库中的std::thread,这取决于开发者的选择。无论哪种方式,新创建的线程都会运行一个函数,该函数负责读取客户端...

    FFmpegSDL 音视频开发 ④ ( SDL 多线程 / 创建线程 / 等待线程执行完毕 )

    【FFmpeg】SDL 音视频开发 ④ ( SDL 多线程 | SDL_Thread 结构体 | SDL_CreateThread 函数创建线程 | SDL_WaitThread 函数等待线程执行 ) https://hanshuliang.blog.csdn.net/article/details/139760886 博客源码...

    Win32 多线程程序设计_纯文本转成的PDF

    1. **线程基础知识**:首先会介绍什么是线程,线程与进程的区别,以及为何在某些场景下选择使用多线程。线程共享内存空间,这带来了数据通信的便利,但同时也引入了线程同步和互斥的问题。 2. **线程创建和销毁**:...

    MFC线程创建之旅.pdf

    3. `CWinThread::CreateThread`使用_CRT的`_beginthreadex`函数创建线程,`_beginthreadex`是Windows API `CreateThread`的封装。 4. `_beginthreadex`内部调用了`_threadstartex`,这是一个线程启动函数,它使用了...

Global site tag (gtag.js) - Google Analytics