C++封装一个简单的线程类

多线程编程简介:

大家在编程时,经常需要在程序中启动一个或多个线程来处理任务,而如果每次都是去调用系统创建线程的API函数来创建,代码量虽不多,但线程的创建和业务逻辑代码就揉在一起了,且创建多个线程的时候,有大量的重复代码,不便于维护。若我们把创建线程和销毁线程的这些共同的代码封装到一个类中,这样我们可以更专注业务逻辑的实现,在其它地方直接拿来用就行,程序也便于维护和扩展。而且这样的话即使程序所使用的线程库更换了,但线程类提供的接口没变,所以我们的业务逻辑代码也不用任何的改动。

创建一个线程也无非就是调用系统线程API或第三方库的API,然后传入线程函数地址和线程运行所需要的参数即可,所以我们需要将此部分代码抽象出来,并提供使用接口即可。

一个线程基类Thread

这里我们使用的是Apache提供的apr库,该库具有跨平台性,当然不管使用什么库,我们提供的接口都是一样的,且线程创建和销毁的逻辑是一样的。代码:

 1 class Thread

 2 {

 3 public:

 4     Thread(bool bDetach = true);

 5     virtual ~Thread();

 6 

 7     virtual void run() = 0;      //业务接口

 8 

 9     int       start();              //启动线程

10     int      join();                //等待线程线束

11     void    destroy();           //销毁线程所申请的资源

12 

13     // attribute functions

14     int      get_thread_id()        { return thr_id_; }

15     void    set_thread_id(unsigned long thrId) {    thr_id_ = thrId; }

16 

17 protected:

18     static    void* __stdcall thread_proc(apr_thread_t* th, void* data);

19     void    notify() { cond_.signal(); }

20     bool    check_interrupt() { return bExit_; }

21 

22 private:

23     size_t            thr_id_;        //线程ID

24     bool            bExit_;            //线程是否要退出标志

25 

26     apr_thread_t*    thr_;            //线程句柄

27     Condition        cond_;            //线程函数中等待任务的条件变量

28     

29 private:

30     //not implement

31     Thread(const Thread& );

32     Thread& operator=(const Thread& );

33 };

一些说明:

  1. 我们在start()方法中调用apr库提供的线程API创建一个线程: apr_thread_create(),并将线程函数thread_proc()和Thread*为线程函数参数传入apr_thread_create()即可,具体代码在后面贴出。
  2. Join()函数用于等待线束线程,而destroy() 则是用于显示销毁该线程所占用的资源。
  3. 线程基类有一个纯虚函数run(),即应用线程继承Thread类后必须实现run()函数,即实现程序的业务逻辑
  4. 在start()创建完线程后系统便在某一时刻开始执行thread_proc()方法,我们在该方法中会调用run()函数,由于多态性,也就会调用应用程序多实现的run()函数了

具体实现(Thread.cpp):

 1 int    Thread::start()

 2 {

 3     apr_status_t        rv;

 4     apr_threadattr_t*    thrattr = NULL;    

 5     apr_threadattr_create(&thrattr, g_mpool);

 6 

 7     //创建一个线程

 8     if ((rv = apr_thread_create(&thr_, thrattr, Thread::thread_proc, this, g_mpool)) != APR_SUCCESS)

 9     {

10         set_error_code(rv);

11         char errbuf[512];

12         apr_strerror(rv, errbuf, sizeof(errbuf));

13         log2DebugView("Thread: failed create a thread: [%d][%s]\n", rv, errbuf);

14         return rv;

15     }

16     apr_sleep(100000);        //ensure the thead_proc is running

17 

18     return rv;

19 }

20 //等待线束线程

21 int    Thread::join()

22 {

23     bExit_    = true;

24     notify();

25     apr_sleep(100000);

26 

27     apr_status_t rv = 0;

28     return apr_thread_join(&rv, thr_);

29 }

30 //销毁线程

31 void Thread::destroy()

32 {

33     if (!bExit_)

34         join();

35     cond_.destroy();

36 }

37 //线程函数,将会调用子类实现的run()方法

38 void* Thread::thread_proc(apr_thread_t* th, void* data)

39 {

40     Thread* pthis = static_cast<Thread*>(data);

41     while (!pthis->bExit_)

42     {

43         //调用子类实现的run()方法

44         pthis->run();

45 

46         //wait for signal

47         pthis->cond_.wait();

48     }

49 

50     printf("thread exit, id: %d\n", pthis->get_thread_id());

51     apr_thread_exit(th, APR_SUCCESS);

52     return NULL;

53 }

这里我们不要太过意研究线程在具体代码是如何创建的,比如在start()函数中,在windows下线程函数可以是 UINT thread_proc(LPVOID param); 而创建线程则是调用__beginthreadex()的windows API即可,具体可参照windows的线程创建和销毁逻辑。线程使用如下:

应用示例

 1 //继承Thread类并实现run()接口,有点类似Java或C#的用法

 2 class MyThread : public Thread

 3 {

 4 public:

 5     MyThread(){ loop_ = true; }

 6     virtual MyThread(){}

 7 

 8     //只关心如何实现业务逻辑,而看不到线程是如何创建的

 9     virtual void run()

10     {

11         while (loop)

12         {

13             //do some work

14         }

15         printf("MyThread exit.\n");

16     }

17 

18 private:

19     bool loop_;

20 };

21 

22 // 在程序中使用如下

23 MyThread* pmt = new MyThread();

24 pmt->start();        //调用start()方法后,即启动了一个线程了

这样,我们就完成了一个线程类的封装和使用了,代码不多,但很常用哈。最后说明一下线程类中使用一个Condition的类,其实也就是一个对事件的封装使用,完全可以用windows下的 SetEvent()/WaitForSingleObject()替代或Linux下的pthread_condition_t的pthrad_condition_signal()/pthread_condition_wait()替代,即等待事件和通知事件的处理。

    下一节我将会利用这个线程类实现一个简单的线程池,便于我们在程序中使用。

【C/C++开发】C++实现简单的线程类的更多相关文章

  1. Delphi线程类 DIY(把类指针作为参数传进去,就可以执行类里面的方法啦)

    Delphi 封装了一个很强大的线程类 TThread, 我们也自己动手制作一个简单的线程类 首先Type一个类 type TwwThread = class constructor Create;  ...

  2. 【C/C++开发】C++实现简单的线程池

    C++实现简单的线程池 线程池编程简介: 在我们的服务端的程序中运用了大量关于池的概念,线程池.连接池.内存池.对象池等等.使用池的概念后可以高效利用服务器端的资源,比如没有大量的线程在系统中进行上下 ...

  3. iOS开发网络篇—简单介绍ASI框架的使用

    iOS开发网络篇—简单介绍ASI框架的使用 说明:本文主要介绍网络编程中常用框架ASI的简单使用. 一.ASI简单介绍 ASI:全称是ASIHTTPRequest,外号“HTTP终结者”,功能十分强大 ...

  4. 封装一个简单好用的打印Log的工具类And快速开发系列 10个常用工具类

    快速开发系列 10个常用工具类 http://blog.csdn.net/lmj623565791/article/details/38965311 ------------------------- ...

  5. java开发两年,这些线程知识你都不知道,你怎么涨薪?

    前言 什么是线程:程序中负责执行的哪个东东就叫做线程(执行路线,进程内部的执行序列),或者说是进程的子任务. Java中实现多线程有几种方法 继承Thread类: 实现Runnable接口: 实现Ca ...

  6. iOS开发UI篇—简单的浏览器查看程序

    iOS开发UI篇—简单的浏览器查看程序 一.程序实现要求 1.要求 2. 界面分析 (1) 需要读取或修改属性的控件需要设置属性 序号标签 图片 图片描述 左边按钮 右边按钮 (2) 需要监听响应事件 ...

  7. iOS开发UI篇—简单介绍静态单元格的使用

    iOS开发UI篇—简单介绍静态单元格的使用 一.实现效果与说明 说明:观察上面的展示效果,可以发现整个界面是由一个tableview来展示的,上面的数据都是固定的,且几乎不会改变. 要完成上面的效果, ...

  8. iOS开发Swift篇—简单介绍

    iOS开发Swift篇—简单介绍 一.简介 Swift是苹果于2014年WWDC(苹果开发者大会)发布的全新编程语言 Swift在天朝译为“雨燕”,是它的LOGO 是一只燕子,跟Objective-C ...

  9. 李洪强iOS开发之- 实现简单的弹窗

     李洪强iOS开发之- 实现简单的弹窗 实现的效果:  112222222222223333333333333333

随机推荐

  1. React重置非受控组件state的方法

    如果想通过props来重置state的值.有3种方法: 1. 最好的方法:key属性 修改key属性的值,可以使组件卸载后重新加载.所有的状态全部重置. 这种情况可以给key设一个每次渲染都会改变的值 ...

  2. Devtool-Console

    1. console面板展示 1.全屏展示 打开开发者工具(option+cmd+i),点击console的tab 2. 在其他面板展示的同时展示console面板 1)esc快捷命令 2)或者选择打 ...

  3. sql server 悲观锁和乐观锁的作用

    sql server对并发的处理-乐观锁和悲观锁 假如两个线程同时修改数据库同一条记录,就会导致后一条记录覆盖前一条,从而引发一些问题. 例如: 一个售票系统有一个余票数,客户端每调用一次出票方法,余 ...

  4. 模拟I2C协议学习点滴之程序相关定义

    由于主机和从机都会给数据线SDA发信号,比如主机先给SDA发送数据后,从机收到数据后发送应答信号将SDA拉低,故SDA类型设定为inout.而DATA设定为inout类型,是起到校验通信的作用(后续的 ...

  5. LibreOJ #6. Guess Number

    二次联通门 : LibreOJ #6. Guess Number /* LibreOJ #6. Guess Number 交互题初体验 用了二分判定 感觉不错 */ #include "in ...

  6. C++标准库分析总结(三)——<迭代器设计原则>

    本节主要总结迭代器的设计原则,以及iterstor traits的设计作用 1.迭代器遵循的原则 迭代器是算法和容器的桥梁,它是类模板的设计,迭代器必须有能力回答算法提出的问题才能去搭配该算法的使用 ...

  7. C# 获取网络文件 批量压缩成 文件流 并下载 压缩包

    需要的DLL : ICSharpCode.SharpZipLib.dll JS部分 //下载所有文件的 压缩包 function DownAllFile() { //zip文件名 var zipNam ...

  8. 针对于linux初学者的学习(摘自网络端)

    一. 选择适合自己的Linux发行版谈到linux的发行版本,太多了,可能谁也不能给出一个准确的数字,但是有一点是可以肯定的,linux正在变得越来越流行, 面对这么多的Linux 发行版,打算从其他 ...

  9. 读取文件名.cpp

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

  10. Python 自学笔记(六)

    PK小游戏 1.要有玩家敌人:那就是需要定义两个角色的属性变量 2.相互攻击:需要两个角色都有血量和攻击的变量(也就是四个变量) 3.攻击减少血量:比如玩家血量=敌人攻击力-玩家当前血量 4.最终胜负 ...