C++ 多线程2 beginthread 启动线程知识 20131021

Reference: http://blog.csdn.net/laoyang360/article/details/7720656

前言:

之前曾经使用过WINAPI创建线程并且启动的知识,还有线程之间同步的知识,这里讲解一下使用__beginthread的原理和机制。

1._beginthread 简单的介绍

如果我们编写的是C++的程序,不应该使用CreateThread WINAPI接口去创建线程,而是应该使用Visual C++运行库函数_beginthread,退出线程的时候应该使用_endthread。因为_beginthreaex和_endthreadex是CRT的线程函数,所以必须注意编译选项runtimelibrary,使用的是MT或者是MTD(MultiThreaded,debug Multithread);_beginthread函数的参数列表和CreateThread的参数列表完全相同,但是参数名称和类型不是完全的相同。因为Microsoft的C/C++运行库的开的小组认为C/C++运行期函数不应该对Windows数据类型有任何的依赖。

每一个线程都会获得由C/C++运行期库的堆栈分配自己的tiddata内存结构;传递给_beginthreadex的线程函数的地址保存在tiddata内存数据块,传递给该函数的参数也保存在该数据块中;_beginthreadex确实从内部调用CreateThread函数,因为这是创建线程早OS层面上的唯一的方法;当调用CreateThread函数的时候,他会被告知通过调用_threadstartex而不是pfnStartAddr来启动执行新的线程,还有传递给线程函数的从哪回溯是tiddata结构而不是pvParam的地址;如果一切顺利的话,就会像CreateThread那样返回线程句柄,任何操作失败就会返回NULL。

总的来说就是_beginthreadex在内部调用CreateThread函数,在调用之前_beginthreadex做了很多的工作,从而比CreateThread更加安全.

#include <iostream>

#include <string>

#include <process.h>

#include <Windows.h>

using namespace std;

class ThreadX{

private:

int loopStart;

int loopEnd;

int dispFrequency;

public:

string threadName;

ThreadX(int startVal, int endVal, int frequency){

this->loopStart = startVal;

this->loopEnd = endVal;

this->dispFrequency = frequency;

}

static unsigned __stdcall ThreadStaticEntryPoint(void * pThis){

ThreadX* pThX = (ThreadX*) pThis;

pThX->ThreadEntryPoint();

return 1;

}

void ThreadEntryPoint(){

for (int i = loopStart; i <= loopEnd; i++){

cout << threadName << " i = " << i << endl;

Sleep(100);

}

}

};

int main(){

ThreadX * pThX = new ThreadX(0, 10, 2000);

HANDLE hth1 = NULL;

unsigned uiThread1ID;

hth1 =(HANDLE) _beginthreadex(NULL,

0,

ThreadX::ThreadStaticEntryPoint,

pThX,

CREATE_SUSPENDED,

&uiThread1ID);

if (hth1 == NULL){

cout << "failed to create thread 1" << endl;

}

DWORD dwExitCode;

GetExitCodeThread(hth1, &dwExitCode); // shoule be STILL_ACTIVE

cout << "init thread 1 exit code id = " << dwExitCode << endl;

pThX->threadName = "yang1";

ThreadX* pThX2 = new ThreadX(0, 10, 2000);

HANDLE hth2 = NULL;

unsigned uiThread2ID;

hth2 = (HANDLE) _beginthreadex(NULL,

0,

ThreadX::ThreadStaticEntryPoint,

pThX2,

CREATE_SUSPENDED,

&uiThread2ID);

if (hth2 == NULL){

cout << "create thread 2 failed" << endl;

}

GetExitCodeThread(hth2, &dwExitCode);

cout << "init thread 2 exit code id = " << dwExitCode << endl;

pThX2->threadName = "yang2";

ResumeThread(hth1);

ResumeThread(hth2);

WaitForSingleObject(hth1, INFINITE);

WaitForSingleObject(hth2, INFINITE);

GetExitCodeThread(hth1, &dwExitCode);

cout << "thread 1 exited with exit code " << dwExitCode << endl;

GetExitCodeThread(hth2, &dwExitCode);

cout << "thread 2 exited with exit code " << dwExitCode << endl;

CloseHandle(hth1);

CloseHandle(hth2);

delete pThX;

delete pThX2;

pThX = NULL;

pThX2 = NULL;

cout << "end program  " << endl;

return 0;

}

2.线程临界区

线程之间共享的对象,当修改的时候,只能够一个线程修改,同时不存在其他的线程访问该资源,这就是线程之间的同步问题.在上一篇总结中使用的是CreateMutex函数创建一个HANDLE,这样的话创建一个Mutex,当线程访问资源的时候,使用WaitForSimpleObject(handl,  INFINITE);获得该对象的锁,然后再使用完的时候,释放对象锁,ReleaseMutex(handle);

在线程中有一个概念是临界区,在C++中使用这一种实现的,就是用CRITICAL_SECTION 声明一个临界区变量,然后再程序开始的时候,使用InitializeCriticalSelection(CRITICAL_SELECTION*) 初始化临界变量,然后当线程进入临界区的时候,使用EneterCriticalSelection(CRITICAL_SELECTION*);当访问完成的时候使用LeaveCriticalSelection(CRITICAL_SELECTION);

#include <iostream>

#include<Windows.h>

using namespace std;

bool g_bContinue = true;

int g_count1 = 0;

int g_count2 = 0;

DWORD WINAPI ThreadProc(LPVOID lpParam){

for (int i = 0; i < 20; i++)

{

EnterCriticalSection(&g_cs);

g_count1++;

g_count2++;

LeaveCriticalSection(&g_cs);

}

return 0;

}

int main(){

HANDLE g_hThr[2];

InitializeCriticalSection(&g_cs);

g_hThr[0] = CreateThread(NULL, 0, ThreadProc, NULL, 0, NULL);

g_hThr[1] = CreateThread(NULL, 0, ThreadProc, NULL, 0, NULL);

WaitForSingleObject( g_hThr[0], INFINITE);

WaitForSingleObject( g_hThr[1], INFINITE);

g_bContinue = false;

CloseHandle(g_hThr[0]);

CloseHandle(g_hThr[1]);

DeleteCriticalSection(&g_cs);

Sleep(1000);

cout << "g_count1 = " << g_count1 << "\t g_count2 = " << g_count2 << endl;

}

追梦的飞飞

于广州中山大学 20131022

HomePage: http://yangtengfei.duapp.com

转载于:https://www.cnblogs.com/hbhzsysutengfei/p/3409483.html

C++多线程2.beginthread相关推荐

  1. 按键精灵定义全局变量_按键精灵2014全局数组变量传递(支持静态与动态)V2

    马上注册,结交更多好友,享用更多功能,让你轻松玩转社区. 您需要 登录 才可以下载或查看,没有帐号?立即注册 x 最新版按键精灵2014已支持数组全局变量了,也就意味着以后多线程不需要再借助插件或字符 ...

  2. 【java的socket编程】结合多线程Thread实现通信(使用线程池和非线程池对比)、java开发UDP/IP网络程序

    结合多线程实现socket 使用非线程池(拓展Thread) 使用线程池(Executor pool) 使用DatagramPacket DatagramSocket开发UDP/IP程序 使用UDP获 ...

  3. BCB 多线程的同步与协调

    多线程编程是提高系统资源利用率的一种常见方式.它占用的资源更小,启动更快,还可以实现在后台运行一些需时较长的操作.[喝小酒的网摘]http://blog.hehehehehe.cn/a/8498.ht ...

  4. Java多线程知识小抄集(一)

    欢迎支持笔者新作:<深入理解Kafka:核心设计与实践原理>和<RabbitMQ实战指南>,同时欢迎关注笔者的微信公众号:朱小厮的博客. 欢迎跳转到本文的原文链接:https: ...

  5. delphi 中几种多线程操作方式

    在了解多线程之前我们先了解一下进程和线程的关系 一个程序至少有一个主进程,一个进程至少有一个线程. 主线程又程为UI线程. 进程和线程的主要差别在于它们是不同的操作系统资源管理方式.进程有独立的地址空 ...

  6. C++11 多线程 基础

    C++11开始支持多线程编程,之前多线程编程都需要系统的支持,在不同的系统下创建线程需要不同的API如pthread_create(),Createthread(),beginthread()等,使用 ...

  7. Delphi多线程详解CreateThread、TThread,以及线程间通过临界区(CriticalSection)实现同步

     在了解多线程之前我们先了解一下进程和线程的关系 一个程序至少有一个主进程,一个进程至少有一个线程. 为了保证线程的安全性请大家看看下面介绍 Delphi多线程同步的一些处理方案大家可以参考:ht ...

  8. delphi 多线程3

     多线程程序设计 我们知道,win95或winNT都是"多线程"的操作系统,在DELPHI 2.0中,我们可以充分利用这一特性,编写出"多线程"的应用程序. 对 ...

  9. [Delphi] 多线程编程

    本篇简单介绍一下Delphi下多线程的编写 一.CreateThread 这个是最原始最基础的方法,利用Windows API来创建 function ThreadProc(param: LPVOID ...

最新文章

  1. 最新Java中Date类型详解
  2. 2019年最新深度强化学习PPT(DeepMind制作)
  3. RabbitMQ接触(二)
  4. 【51NOD - 1523】 非回文(dfs)
  5. Yum 安装 mysql5.7
  6. 在simulink中建立调度增益控制系统的模型
  7. 推荐 | 一个超好的OpenCV4学习社区
  8. 【语音合成】基于matlab线性预测系数和预测误差语音合成【含Matlab源码 564期】
  9. SSM基于web的教务管理系统 毕业设计-附源码261620
  10. html怎么安装系统,一键重装系统_MBR分区一键安装系统教程-爱纯净
  11. 生活污水是怎么处理的
  12. Cocos Creator 热更新文件MD5计算和需要注意的问题
  13. No suitable application records were found. Verify your bundle identifi
  14. 压缩感知算法_CEALeti研发出基于压缩感知的CMOS图像传感器
  15. 策略模式与工厂模式的区别
  16. 京东推了两款无人店产品,是“狼来了”还是“纸老虎”?
  17. C#趣味消消乐小游戏
  18. 面向对象设计思想-基本概念
  19. Unity UI学习(1)——制作技能冷却图标
  20. 警惕!李鬼出没冒充SwapX,千万别上当

热门文章

  1. 一文读懂SCADA、DCS与PLC之间的不同
  2. java hessian 协议_dubbo的Hessian协议的使用
  3. srs之深入浅出看流媒体
  4. 用友华表Cell一些用法小结(cs.net版本)
  5. Clint Eastwood的老爷车
  6. 利用DoraOS将旧PC改造成瘦客户机
  7. 表情包GIF用什么软件 如何制作动态图
  8. 使用WinRAR制作免安装exe文件
  9. 5款不花钱的国产良心软件,错过真的太可惜
  10. layUI时间选择框和日期间隔