文档章节

c++多线程编程之CreateThread与_beginthreadex本质区别

moki_oschina
 moki_oschina
发布于 2014/01/02 21:13
字数 2468
阅读 577
收藏 4
点赞 0
评论 0

使用多线程其实是非常容易的,下面这个程序的主线程会创建了一个子线程并等待其运行完毕,子线程就输出它的线程ID号然后输出一句经典名言——Hello World。整个程序的代码非常简短,只有区区几行。

//最简单的创建多线程实例
#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。很多运行库中的函数在出错时会将错误代号赋值给这个全局变量,这样可以方便调试。但如果有这样的一个代码片段:

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()函数的区别。

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

接下来,类似于上面的程序用CreateThread()创建输出“Hello World”的子线程,下面使用_beginthreadex()来创建多个子线程:

//创建多子个线程实例

#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,……。要实现这个功能似乎非常简单——每个子线程对一个全局变量进行递增并输出就可以了。代码如下:

//子线程报数
#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;
}

 

 

本文转载自:http://blog.csdn.net/morewindows/article/details/7421759

共有 人打赏支持
moki_oschina
粉丝 24
博文 179
码字总数 22482
作品 0
成都
程序员
Win32 多线程的创建方法,区别和联系

Win32多线程的创建方法主要有: CreateThread() beginthread()&&beginthreadex() AfxBeginThread() CWinThread类 一、简介 CreateThread: Win32提供的创建线程的最基础的API,用于在主线程上创...

山里来的鱼
2015/07/21
0
0
_beginThreadex创建多线程解读

_beginThreadex创建多线程解读 _beginThreadex创建多线程解读 一、需要的头文件支持 #include <process.h> // for _beginthread() 需要的设置:ProjectàSetting-->C/C++-->User run-time lib......

IMGTN
2012/11/25
0
0
【转】_beginthread()与CreateThread()的区别

一个时运行时库函数,一个时WIN32 的库函数。 CreateThread、beginthread和beginthreadex都是用来启动线程的,但大家看到oldworm没有提供beginthread的方式,原因简单,beginthread是beginth...

zxs2015
2013/01/08
0
2
windows多线程详解

在一个牛人的博客上看到了这篇文章,所以就转过来了,地址是http://blog.csdn.net/morewindows/article/details/7421759 本文将带领你与多线程作第一次亲密接触,并深入分析CreateThread与b...

长平狐
2013/12/25
645
0
如何自学成为C/C++程序员大牛

C语言是面向过程的,而C++是面向对象的 C和C++的区别: C是一个结构化语言,它的重点在于算法和数据结构。C程序的设计首要考虑的是如何通过一个过程,对输入(或环境条件)进行运算处理得到...

小辰带你看世界
05/11
0
0
秒杀多线程第二篇 多线程第一次亲密接触 CreateThread与_beginthreadex本质区别

本文将带领你与多线程作第一次亲密接触,并深入分析CreateThread与beginthreadex的本质区别,相信阅读本文后你能轻松的使用多线程并能流畅准确的回答CreateThread与beginthreadex到底有什么区...

晨曦之光
2012/05/21
109
0
秒杀多线程第二篇 多线程第一次亲密接触 CreateThread与_beginthreadex本质区别

本文将带领你与多线程作第一次亲密接触,并深入分析CreateThread与beginthreadex的本质区别,相信阅读本文后你能轻松的使用多线程并能流畅准确的回答CreateThread与beginthreadex到底有什么区...

彭博
2012/04/12
192
0
C语言/C++对编程学习的重要性!

C语言是面向过程的,而C++是面向对象的 C和C++的区别: C是一个结构化语言,它的重点在于算法和数据结构。C程序的设计首要考虑的是如何通过一个过程,对输入(或环境条件)进行运算处理得到...

小辰带你看世界
06/02
0
0
秒杀多线程第二篇 多线程第一次亲密接触 CreateThread与_beginthreadex本质区别

本文将带领你与多线程作第一次亲密接触,并深入分析CreateThread与beginthreadex的本质区别,相信阅读本文后你能轻松的使用多线程并能流畅准确的回答CreateThread与beginthreadex到底有什么区...

长平狐
2012/12/10
20
0
error LNK2001 unresolved external symbol _Wi...

学习VC++时经常会遇到链接错误LNK2001,该错误非常讨厌,因为对于编程者来说,最好改的错误莫过于编译错误,而一般说来发生连接错误时,编译都已通过。产生连接错误的原因非常多,尤其LNK...

huanongkou
2013/03/21
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

shell中的函数、shell中的数组、告警系统需求分析

shell中的函数 格式: 格式: function f_name() { command } 函数必须要放在最前面 示例1(用来打印参数) 示例2(用于定义加法) 示例3(用于显示IP) shell中的数组 shell中的数组1 定义数...

Zhouliang6
今天
2
0
用 Scikit-Learn 和 Pandas 学习线性回归

      对于想深入了解线性回归的童鞋,这里给出一个完整的例子,详细学完这个例子,对用scikit-learn来运行线性回归,评估模型不会有什么问题了。 1. 获取数据,定义问题     没有...

wangxuwei
今天
1
0
MAC安装MAVEN

一:下载maven压缩包(Zip或tar可选),解压压缩包 二:打开终端输入:vim ~/.bash_profile(如果找不到该文件新建一个:touch ./bash_profile) 三:输入i 四:输入maven环境变量配置 MAVEN_HO...

WALK_MAN
今天
0
0
33.iptables备份与恢复 firewalld的9个zone以及操作 service的操作

10.19 iptables规则备份和恢复 10.20 firewalld的9个zone 10.21 firewalld关于zone的操作 10.22 firewalld关于service的操作 10.19 iptables规则备份和恢复: ~1. 保存和备份iptables规则 ~2...

王鑫linux
今天
2
0
大数据教程(2.11):keeperalived+nginx高可用集群搭建教程

上一章节博主为大家介绍了目前大型互联网项目的系统架构体系,相信大家应该注意到其中很重要的一块知识nginx技术,在本节博主将为大家分享nginx的相关技术以及配置过程。 一、nginx相关概念 ...

em_aaron
今天
1
0
Apache Directory Studio连接Weblogic内置LDAP

OBIEE默认使用Weblogic内置LDAP管理用户及组。 要整理已存在的用户及组,此前办法是导出安全数据,文本编辑器打开认证文件,使用正则表达式获取用户及组的信息。 后来想到直接用Apache Dire...

wffger
今天
2
0
HFS

FS,它是一种上传文件的软件。 专为个人用户所设计的 HTTP 档案系统 - Http File Server,如果您觉得架设 FTP Server 太麻烦,那么这个软件可以提供您更方便的档案传输系统,下载后无须安装,...

garkey
今天
1
0
Java IO类库之BufferedInputStream

一、BufferedInputStream介绍 /** * A <code>BufferedInputStream</code> adds * functionality to another input stream-namely, * the ability to buffer the input and to * sup......

老韭菜
今天
0
0
STM 32 窗口看门狗

http://bbs.elecfans.com/jishu_805708_1_1.html https://blog.csdn.net/a1985831055/article/details/77404131...

whoisliang
昨天
1
0
Dubbo解析(六)-服务调用

当dubbo消费方和提供方都发布和引用完成后,第四步就是消费方调用提供方。 还是以dubbo的DemoService举例 -- 提供方<dubbo:application name="demo-provider"/><dubbo:registry address="z...

青离
昨天
2
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部