为什么要使用ACE_Task来管理线程

从C#转到C++后,感觉到C++比C#最难的地方,就是在系统编程时,C#中有对应的类库,我接触到一个类后,就可以通过这个类,知道很多相关的功能。而在C++中,必须使用Windows API,但是Windows API的缺点反而是不系统。比如,想要创建一个线程时,C#中可以通过新建一个Thread类,我得到这个类后,就可以通过帮助或者MSDN了解到这个类中有哪些方法,如何管理线程等。但是C++中,创建一个线程,需要调用CreateThread,但因为它是一个API,所以很难快速的了解到与线程管理相关的API,至少对我来说,这是一个难点。

C#的优势是以类的方式去管理线程,而C++ 调用API更像是在单打独斗,显得特别特别散乱。

那么在C++中,我们能不能以类的方式去管理线程呢。针对这个问题,ACE_Task便浮出了水面。

ACE_Task使用方法

ACE_Task对常用线程处理进行了OO包装,通过ACE_Task,能对线程进行更好的操作。

根据ACE_Task的父类ACE_Task_Base中的注释,可以总结出以下几点:

  • 想要使用ACE_Task,就要从ACE_Task派生一个子类,然后实现比较重要的几个虚方法。
  • 实现服务初始化和终止方法:open()方法应该包含所有专属于任务的初始化代码。其中可能包括诸如连接控制块、锁和内存这样的资源。close() 方法是相应的终止方法。
  • 调用启用(Activation )方法:在主动对象实例化后,你必须通过调用activate()启用它。要在主动对象中创建的线程的数目,以及其他一些参数,被传递给activate()方法。activate()方法会使svc()方法成为所有它生成的线程的启动点。
  • 实现服务专有的处理方法:如上面所提到的,在主动对象被启用后,各个新线程在svc()方法中启动。应用开发者必须在子类中定义此方法。

编写ACE_Task的Demo

下面就通过总结的几点来编写一个Demo

首先是派生ACE_Task的子类ShowTask

#pragma once
#include "ace/Task_T.h"
#include <string>
#include <iostream>
using namespace std; class ShowTask : public ACE_Task<ACE_SYNCH>
{
public:
ShowTask(const char* str_to_show, int interval) : show_str_(str_to_show), interval_(interval)
{
//启动线程
activate();
}
~ShowTask()
{
//结束线程
close();
}
public:
virtual int open( void *args = 0 )
{
cout << "线程启动" << endl;
return 0;
} virtual int close( u_long flags = 0 )
{
cout << "线程结束" << endl;
return 0;
} virtual int svc( void )
{
int time_begin = GetTickCount();
while(true)
{
int time_end = GetTickCount();
if(time_end - time_begin >= interval_)
{
time_begin = time_end;
cout << show_str_ << endl;
}
Sleep(10);
}
return 0;
}
private:
//需要显示的信息
string show_str_;
//间隔多长时间进行显示
int interval_;
};

然后在程序中调用ShowTask

#include "ShowTask.h"

int _tmain(int argc, _TCHAR* argv[])
{
ACE::init();
string str = "这是一个测试程序";
ShowTask task(str.c_str(), 1000); Sleep(10000);
ACE::fini();
return 0;
}

运行效果图:

需要注意的地方

在程序中使用ACE_Task,那么就必须在使用前对ACE进行初始化,即调用ACE::init方法,然后在程序结束的时候调用ACE::fini方法。如果没有进行初始化就调用ShowTask,那么程序就会崩溃。

至于为什么会崩溃,以及ACE::init方法做了些什么,后续的文章中会详细的介绍。

关于open方法的疑惑

从上面的测试图中可以看到,线程结束后,调用了close方法,但是在线程启动时并没有调用open方法。

而且从谷歌找到的资料中也介绍,在线程启动的时候回调用open方法,而且他们的示例程序与本文中提供的demo中关于ACE_Task的使用方法也基本一致。

那么open方法为什么没有被调用呢,究竟是使用方法不对,还是因为ACE版本不同导致的。关于这个问题也会在后续的文章中进行详细的介绍。

使用ACE_Task管理线程的更多相关文章

  1. 使用Executor管理线程

    上一篇博客(第一个并发程序:定义任务和驱动任务)中,我们使用Thread对象启动线程,而java.util.concurrent包的Executor执行器提供了更好的管理Thread对象的方法,从而简 ...

  2. java笔记--用ThreadLocal管理线程,Callable<V>接口实现有返回值的线程

    用ThreadLocal管理线程,Callable<V>接口实现有返回值的线程 ThreadLocal在我的笔记"关于线程同步"的第5种方式里面有介绍,这里就不多说了. ...

  3. Java基础之线程——管理线程同步方法(BankOperation2)

    控制台程序. 当两个或多个线程共享同一资源时,例如文件或内存块,就需要采取措施,确保其中的一个线程不会修改另一个线程正在使用的资源.当其中的一个线程更新文件中的某个记录,同时另一个线程正在检索这个记录 ...

  4. C# 多线程的自动管理(线程池) 基于Task的方式

    C# 多线程的自动管理(线程池) 在多线程的程序中,经常会出现两种情况:    1. 应用程序中线程把大部分的时间花费在等待状态,等待某个事件发生,然后给予响应.这一般使用 ThreadPool(线程 ...

  5. 学习pthreads,管理线程的栈

    进程的地址空间分成代码段,静态数据段,堆和栈段.线程栈的位置和大小是从它所属的进程的栈中切分出来的.每个栈必须足够大,以容纳所有对等线程的函数的执行以及它们将会调用的例程链.或许你会问为什么要进行线程 ...

  6. C#线程篇---线程池如何管理线程(6完结篇)

    C#线程基础在前几篇博文中都介绍了,现在最后来挖掘一下线程池的管理机制,也算为这个线程基础做个完结. 我们现在都知道了,线程池线程分为工作者线程和I/O线程,他们是怎么管理的? 对于Microsoft ...

  7. 为什么使用 Executor 框架比使用应用创建和管理线程好?

    为什么要使用 Executor 线程池框架 1.每次执行任务创建线程 new Thread()比较消耗性能,创建一个线程是比较耗 时.耗资源的. 2.调用 new Thread()创建的线程缺乏管理, ...

  8. 使用ExecutorCompletionService 管理线程池处理任务的返回结果

    在我们日常使用线程池的时候,经常会有需要获得线程处理结果的时候.此时我们通常有两种做法. 1. 使用并发容器将callable.call() 的返回Future存储起来.然后使用一个消费者线程去遍历这 ...

  9. [Swift实际操作]七、常见概念-(12)使用DispatchGroup(调度组)管理线程数组

    本文将为你演示调度组的使用,使用调度组可以将多个线程中的人物进行组合管理,可以设置当多个相同层次的任务完成之后,再执行另一项任务. 首先导入需要使用的界面工具框架 import UIKit 在控制台输 ...

随机推荐

  1. JSP日期时间转C#

    DateTime.ParseExact("Wed Aug 03 16:46:24 CST 2016", "ddd MMM dd HH:mm:ss CST yyyy&quo ...

  2. Ext开场表单布局设计

    var form = new Ext.form.FormPanel({ labelAlign: 'right', labelWidth: 60, buttonAlign: 'center', titl ...

  3. freeCodeCamp:Slasher Flick

    返回一个数组被截断n个元素后还剩余的元素,截断从索引0开始. /*思路 利用splice截断数组arr,返回运算后的数组arr; 如果howMany大于或等于数组长度,则返回空数组: */ funct ...

  4. html5 canvas绘画时钟

    本示例使用HTML5 canvas,模拟显示了一个时钟, 请使用支持HTML5的浏览器预览效果: HTML部分: <!DOCTYPE html> <html lang="e ...

  5. hdoj 1002 A+B(2)

    Problem Description I have a very simple problem for you. Given two integers A and B, your job is to ...

  6. 【温故Delphi】之VCL消息机制小结

    TObject消息分派 procedure Dispatch(var Message); virtual; #负责分派消息到特定VCL组件的事件处理函数 procedure DefaultHandle ...

  7. priority_queue 示例

    http://www.cplusplus.com/reference/queue/priority_queue/ priority_queue 的top始终保持着为一堆数据中的最大元素. 读取最小 O ...

  8. checkbox、全选反选,获取值

    <input id="Chk_All" onclick="CheckAll()" type="checkbox" /> < ...

  9. ubuntu 安装JAVA jdk的两种方法:

    ubuntu 安装jdk 的两种方式: 1:通过ppa(源) 方式安装. 2:通过官网下载安装包安装. 这里推荐第1种,因为可以通过 apt-get upgrade 方式方便获得jdk的升级 使用pp ...

  10. Encapsulation、Inheritance、Polymorphism

    public class SoldierDemo { public static void main(String[] args) { /*Soldier test1=new Army("张 ...