`

使用 ACE 库框架在 UNIX 中开发高性能并发应用

    博客分类:
  • ACE
阅读更多

使用 ACE 库框架在 UNIX 中开发高性能并发应用


ACE 开放源码工具包可以帮助开发人员创建健壮的可移植多线程应用程序。本文讨论创建使用 ACE 线程的应用程序的一些方法。

Adaptive Communication Environment (ACE) 是一个高性能、开放源码、面向对象的框架和 C++ 类库,它有助于简化网络应用程序的开发。ACE 工具包包括一个操作系统层和一个封装网络 API 的 C++ 外观(facades)集合。本文讨论如何使用 ACE 线程设计高性能、并发、面向对象的网络应用程序。对 ACE 的完整说明,包括如何下载和安装这个工具包,请参见 参考资料。

用于创建和管理线程的 ACE 类

在进程中生成和管理多个线程涉及下面的类:

ACE_Thread_Manager:这是负责创建、管理和同步线程的主要的类。每种操作系统在处理线程方面有细微差异,这个类对应用程序开发人员隐藏这些差异。

ACE_Sched_Params:使用这个类管理各个线程的调度优先级,调度优先级是在 ACE 源代码发行版的 ace/Sched_Params.h 头文件中定义的。可以采用不同的调度策略,可以是 “先到先服务” 的循环方式。

ACE_TSS:在多线程应用程序中使用全局变量会导致同步问题。ACE_TSS 类提供与线程相关的存储模式,可以对那些对于程序是全局的,但是对于每个线程私有的数据提供抽象。ACE_TSS 类提供 operator() 方法,这个方法提供与线程相关的数据。


了解线程管理器类

原生操作系统线程 API 是不可移植的:存在语法和语义差异。例如,UNIX® pthread_create() 和 Windows® CreateThread() 方法都创建线程,但是语法不一样。ACE_Thread_Manager 类提供以下功能:

它可以生成一个或更多线程,每个线程运行自己指定的函数。

它可以作为一个集合(称为 线程组)管理相关的线程。

它管理各个线程的调度优先级。

它允许在线程之间进行同步。

它可以修改线程属性,比如堆栈大小。

表 1 介绍 ACE_Thread_Manager 类的重要方法。


表 1. ACE_Thread_Manager 类的方法

方法名 说明

instance ACE_Thread_Manager 类是一个单实例类,使用这个方法访问线程管理器的惟一实例。

spawn 这个方法创建一个新线程,它的一个输入参数是 C/C++ 函数指针,这个函数执行应用程序的特定工作。

exit 这个方法终止一个线程,释放这个线程的所有资源。

spawn_n 这个方法创建属于同一个线程组的多个线程。

close 这个方法关闭已经创建的所有线程并释放它们的资源。

suspend 线程管理器暂停指定的线程。

resume 线程管理器恢复执行前面暂停的线程。

使用 ACE_Thread_Manager 类的变体

可以作为单实例类使用 ACE_Thread_Manager 类,也可以创建这个类的多个实例。对于单一实例,通过调用 instance 方法访问实例。如果需要管理多个线程组,可以创建不同的线程管理器类,每个类控制它自己的线程集。

清单 1 中的示例创建一个线程。


清单 1. 使用 ACE_Thread_Manager 类创建一个线程

#include "ace/Thread_Manager.h"

#include <iostream>


void thread_start(void* arg)

{

  std::cout << "Running thread..\n";

}


int ACE_TMAIN (int argc, ACE_TCHAR* argv[])

{

  ACE_Thread_Manager::instance()->spawn((ACE_THR_FUNC)thread_start);

  return 0;

}


清单 2 给出 spawn() 方法的原型(取自 ace/Thread_Manager.h)。


清单 2. ACE_Thread_Manager::spawn 方法的原型

int spawn (ACE_THR_FUNC func,

             void *arg = 0,

             long flags = THR_NEW_LWP | THR_JOINABLE | THR_INHERIT_SCHED,

             ACE_thread_t *t_id = 0,

             ACE_hthread_t *t_handle = 0,

             long priority = ACE_DEFAULT_THREAD_PRIORITY,

             int grp_id = -1,

             void *stack = 0,

             size_t stack_size = ACE_DEFAULT_THREAD_STACKSIZE,

             const char** thr_name = 0);


对于初学者来说,创建线程需要的参数数量似乎太多了,所以我们详细讨论一下各个参数和它们的作用:

ACE_THR_FUNC func:这是在生成线程时调用的函数。

void* arg:这是在生成线程时调用的函数的参数。void* 意味着用户可以传递应用程序特有的任何数据类型,甚至可以使用某种结构把多个参数组合成单一数据。

long flags:使用 flags 变量设置生成的线程的几个属性。各个属性都由单一位表示,按照 “或” 关系组合在一起。表 2 说明一些属性。

ACE_thread_t *t_id:使用这个函数访问创建的线程的 ID。每个线程具有惟一的 ID。

long priority:这是生成的线程的优先级。

int grp_id:如果提供这个参数,那么它表示生成的线程是否属于现有的某一线程组。如果传递 -1,那么创建新的线程组并在这个组中添加生成的线程。

void* stack:这是预先分配的堆栈区域的指针。如果提供 0,就请求操作系统提供生成的线程的堆栈区域。

size_t stack_size:这个参数指定线程堆栈的大小(字节数)。如果对于前一个参数(堆栈指针)指定了 0,那么请求操作系统提供大小为 stack_size 的堆栈区域。

const char** thr_name:这个参数只与支持线程命名的平台(比如 VxWorks)相关。对于 UNIX 平台,在大多数情况下忽略它。


表 2. 线程属性及其说明

线程创建标志 说明

THR_CANCEL_DISABLE 不允许取消这个线程。

THR_CANCEL_ENABLE 允许取消这个线程。

THR_DETACHED 创建异步线程。线程的退出状态对于其他任何线程不可用。当线程终止时,操作系统回收线程资源。

THR_JOINABLE 允许新线程的退出状态对于其他线程可用。这也是 ACE 创建的线程的默认属性。当这种线程终止时,操作系统不回收线程资源,直到其他线程联结它为止。

THR_NEW_LWP 创建显式的内核级线程(而不是用户级线程)。

THR_SUSPENDED 创建处于暂停状态的新线程。

清单 3 中的示例使用线程管理器类的 spawn_n 方法创建多个线程。


清单 3. 使用 ACE_Thread_Manager 类创建多个线程 

#include "ace/Thread_Manager.h"

#include <iostream>


void print (void* args)

{

  int id = ACE_Thread_Manager::instance()->thr_self();

  std::cout << "Thread Id: " << id << std::endl;

}


int ACE_TMAIN (int argc, ACE_TCHAR* argv[])

{

  ACE_Thread_Manager::instance()->spawn_n(

      4, (ACE_THR_FUNC) print, 0, THR_JOINABLE | THR_NEW_LWP);


  ACE_Thread_Manager::instance()->wait();

  return 0;

}


ACE 中的另一种线程创建机制

本节讨论 ACE 提供的另一种线程创建/管理机制。这种方法不需要对线程管理器进行显式的细粒度的控制。在默认情况下,每个进程在创建时有一个线程,这个线程在 main 函数开始时启动,在 main 结束时终止。其他线程都需要显式地创建。创建线程的另一种方式是创建预定义的 ACE_Task_Base 类的子类,然后覆盖 svc() 方法。新线程在 svc() 方法中启动,在 svc() 方法返回时终止。在进一步解释之前,请看一下 清单 4 所示的源代码。


清单 4. 使用 ACE_Task_Base::svc 创建线程

#include “ace/Task.h”


class Thread_1 : public ACE_Task_Base { 

  public: 

    virtual int svc( ) { 

       std::cout << “In child’s thread\n”;

       return 0;

    }

 };


int main ( )

   Thread_1 th1;

   th1.activate(THR_NEW_LWP|THR_JOINABLE);

   th1.wait();

   return 0;

}


在 svc() 方法中编写与应用程序相关的线程行为。通过调用 activate() 方法(在 ACE_Task_Base 类中声明和定义)执行线程。在激活线程之后,main() 函数等待子线程完成执行。这就是 wait() 方法的作用:在 Thread_1 执行完之前,主线程被阻塞。这一等待过程是必需的;否则,主线程会调度子线程并执行退出。在看到主线程退出时,C 运行时会销毁所有子线程;因此,子线程可能根本没有被调度或执行。

详细了解 ACE_Task_Base 类

下面详细看看 ACE_Task_Base 中的几个方法。

清单 5 给出 activate() 方法的原型。


清单 5. ACE_Task_Base::activate 方法的原型

virtual int activate (long flags = THR_NEW_LWP | THR_JOINABLE | THR_INHERIT_SCHED,

                        int n_threads = 1,

                        int force_active = 0,

                        long priority = ACE_DEFAULT_THREAD_PRIORITY,

                        int grp_id = -1,

                        ACE_Task_Base *task = 0,

                        ACE_hthread_t thread_handles[ ] = 0,

                        void *stack[ ] = 0,

                        size_t stack_size[ ] = 0,

                        ACE_thread_t thread_ids[ ] = 0,

                        const char* thr_name[ ] = 0);


可以使用 activate() 方法创建一个或多个线程,每个线程调用相同的 svc() 方法,所有线程采用相同的优先级并具有相同的组 ID。下面简要介绍一些输入参数:

long flags:参见 表 2。

int n_threads:n_threads 指定要创建的线程的数量。

int force_active:如果这个标志是 True,而且存在这个任务已经生成的线程,那么新生成的所有线程会共享以前生成的线程的组 ID,忽略传递给 activate() 方法的值。

long priority:这个参数指定线程或线程集合的优先级。调度优先级值是与操作系统相关的,坚持使用默认值 ACE_DEFAULT_THREAD_PRIORITY 是最安全的。

ACE_hthread_t thread_handles:如果 thread_handles 不是零,那么在生成 n 个线程之后,会把各个线程句柄赋值给这个数组。

void* stack:如果指定这个参数,它指定一个指针数组,这些指针指向各个线程的堆栈基。

size_t stack_size:如果指定这个参数,它指定一个整数数组,这些整数表示各个线程堆栈的大小。

ACE_thread_t thread_ids:如果 thread_ids 不是零,那么这个参数是一个数组,其中包含 n 个新生成的线程的 ID。

清单 6 给出 ACE_Task_Base 类中另外几个有用的例程。


清单 6. ACE_Task_Base 中的其他例程

// Block the main thread until all threads of this task are completed 

virtual int wait (void);


// Suspend a task

virtual int suspend (void);


// Resume a suspended task.

virtual int resume (void);


// Gets the no. of active threads within the task

size_t thread_count (void) const;


// Returns the id of the last thread whose exit caused the thread count 

// of this task to 0. A zero return status implies that the result is 

// unknown. Maybe no threads are scheduled.

ACE_thread_t last_thread (void) const;


为了创建处于暂停状态的线程(而不是通过调用 suspend() 方法显式地暂停),需要向 activate() 方法传递 THR_SUSPENDED 标志。可以通过调用 resume() 方法恢复执行线程,见 清单 7。


清单 7. 暂停线程和恢复执行

Thread_1 th1;

th1.activate(THR_NEW_LWP|THR_JOINABLE|THR_SUSPENDED);

…// code in the main thread

th1.resume();

…// code continues in main thread


再看看线程标志

有两种线程:内核级线程和用户级线程。如果不带任何参数调用 activate() 方法,那么默认情况下创建内核级线程。内核级线程与操作系统直接交互,由内核级调度器调度。与此相反,用户级线程在进程范围内运行,为了完成某些任务,根据需要 “分配” 内核级线程。THR_NEW_LWP 标志(activate() 方法的默认参数)总是确保新创建的线程是内核级线程。

线程钩子

ACE 提供一个全局的线程启动钩子,这允许用户执行可以应用于所有线程的任何操作。为了创建启动钩子,需要创建预定义类 ACE_Thread_Hook 的子类并提供 start() 方法定义。start() 接受两个参数:一个用户定义函数的指针和传递给这个用户定义函数的 void*。为了注册钩子,需要调用静态方法 ACE_Thread_Hook::thread_hook,见 清单 8。


清单 8. 使用全局线程钩子

#include "ace/Task.h"

#include "ace/Thread_Hook.h"

#include <iostream>


class globalHook : public ACE_Thread_Hook {

  public:

    virtual ACE_THR_FUNC_RETURN start (ACE_THR_FUNC func, void* arg) {

        std::cout << "In created thread\n";

        (*func)(arg);

    }

};


class Thread_1 : public ACE_Task_Base {

  public:

    virtual int svc( ) {

       std::cout << "In child's thread\n";

       return 0;

    }

 };



int ACE_TMAIN (int argc, ACE_TCHAR* argv[])

{

  globalHook g;

  ACE_Thread_Hook::thread_hook(&g);

  Thread_1 th1;

  th1.activate();

  th1.wait();

  return 0;

}


注意,自动传递给启动钩子的 ACE_THR_FUNC 指针是在执行线程的 activate() 方法时调用的相同函数。以上代码的输出是:

In created thread

In child’s thread


结束语

本文简要讨论了如何使用 ACE 框架创建和管理线程。ACE 框架还有其他一些有用的特性,比如互斥、用于同步的保护阻塞、共享内存和网络服务。详细信息请参见 参考资料。(责任编辑:A6)

分享到:
评论

相关推荐

    ACE开发指南

    ACE还实现了多种基于套接字的高级抽象,如acceptor、connector、以及高层的通信机制如ACE_Reactor、ACE_Service_Configurator等,这些都是ACE框架用来管理并发和多线程通信的核心组件。 5. 文件中还提到了ACE的...

    C++网络编程ACE框架

    ACE(Adaptive Communication Environment)框架是一个开源的、跨平台的网络编程库,主要用于构建高性能、高可用性的分布式系统。ACE提供了一组高级的网络编程接口,简化了多线程、异步I/O、并发控制、对象间通信等...

    ACE网络框架中文文档

    压缩包中的"ACE.chm"文件是ACE的中文帮助文档,包含了详细的技术指南、API参考和示例代码,对于学习和使用ACE框架来说是非常宝贵的资源。通过阅读这个文档,你可以了解到如何配置ACE环境、如何使用其提供的各种组件...

    ACE的中文文档(讲解ACE库)

    1. **ACE概述**:ACE库的起源、设计目标以及它如何帮助开发者构建高性能、高可靠性的网络应用。ACE库的核心组件包括ACE_Win32和ACE_POSIX,分别针对Windows和Unix/Linux等操作系统提供适配层。 2. **ACE的类结构**...

    网络游戏-基于ACE框架的网络视频应用处理系统.zip

    在这个项目中,ACE框架扮演了核心角色,它是Advanced Communication Engine的简称,是一个开源的跨平台通信库,特别适合用于实时、高并发的网络应用开发。 1. ACE框架介绍: ACE框架是AT&T实验室开发的一个C++库,...

    ACE6.1.0库文件

    ACE库,全称为Adaptive ...总的来说,ACE6.1.0为VS2010用户提供了一个强大的网络编程框架,它简化了网络应用开发的复杂性,同时提供了高性能和高度的可移植性。无论是新手还是经验丰富的开发者,都能从中受益。

    ACE开发文档 开发的理想文档

    ACE(Adaptive Communication Environment)是基于C++的跨平台网络编程框架,专为构建高性能、高可用性的分布式系统而设计。作为一个专业的IT开发者,深入理解ACE库能够极大地提升你的网络编程能力,尤其在处理跨...

    ACE-6.3.3最新库

    它提供了一整套跨平台的、高性能的、可扩展的服务,旨在简化分布式系统和网络应用的开发。ACE-6.3.3是这个库的最新版本,它在前一版本的基础上进行了优化和更新,以满足不断变化的开发需求。 在描述中提到,ACE-...

    ACE-5.6.zip ACE5.6官网源码

    ACE(Adaptive Communication Environment)是一个跨平台的C++软件开发框架,主要用于构建高可靠性、高性能的网络通信应用。ACE5.6是该框架的一个版本,提供了丰富的网络编程接口和工具,支持TCP/IP、UDP/IP等多种...

    MinGW版ACE编译好的库(DLL版)

    ACE(Adaptive Communication Environment)是一个跨平台的、开放源代码的C++软件开发框架,用于构建高可靠性、高性能的分布式系统。它提供了大量的网络通信、并发处理和时间管理的类库,广泛应用于嵌入式系统、...

    ACE-6.0.1.rar

    在Linux环境下,用户需要先用tar命令解压,然后可能还需要进行编译和安装步骤,才能在开发项目中使用ACE库。 总的来说,ACE 6.0.1为C++开发者提供了一个功能丰富的网络通信框架,涵盖了从低级网络I/O到高级应用层...

    ACE-6.1.1.tar.gz

    ACE,全称Adaptive Communication Environment,是一个开源的、跨平台的C++库,它提供了用于构建高可靠性、高性能网络应用的工具和框架。ACE的主要设计目标是帮助软件工程师解决分布式系统中的常见问题,如并发控制...

    ACE网络开发.pdf

    ACE(Adaptive Communication Environment)是面向对象的网络通信框架,主要设计用于开发跨平台的高性能通信服务。这个框架提供了一套全面的C++库,帮助开发者处理网络编程中的复杂问题,如并发处理、线程管理、事件...

    基于ACE的FTP服务器

    ACE(Adaptive Communication Environment)是一个跨平台的、开源的C++库,用于构建高性能、高可移植性的网络应用。FTP(File Transfer Protocol)服务器则是用于在网络上进行文件传输的一种服务,它允许用户从远程...

    ACE指南 ACE程序员教程 ACE论文集 ACE应用实例

    ACE是一个强大且灵活的框架,它不仅简化了高性能和实时通信服务及应用的开发过程,还极大地提高了软件的质量和可维护性。通过利用ACE提供的丰富功能和组件,开发者可以更加专注于业务逻辑的实现,而无需担心底层细节...

    ACE最全PPT学习资料

    通过深入学习"ACE最全PPT学习资料",你将能够熟练掌握ACE框架,为你的网络应用开发提供强大的工具支持。在实际工作中,你可以灵活运用ACE来解决复杂的并发和网络问题,提升软件系统的性能和可靠性。

    ACE.rar_ACE

    开发者可以利用ACE提供的组件快速构建高性能、高可用性的通信软件,而无需从零开始编写底层的网络和并发代码。 在深入学习ACE时,"ACE.pdf"这份文档可能会详细阐述ACE的架构、设计原则、使用示例以及如何在项目中...

    ACE-5.4.zip ACE5.4源码

    ACE(Adaptive Communication Environment)是一个跨平台的C++软件框架,专为构建高可靠性、高性能的网络通信应用...在实际项目中,可以结合ACE提供的工具和类,构建复杂的网络通信应用,实现高可靠性和高性能的目标。

    ACE资料_PPT

    2. **ACE框架**:ACE是一个由AT&T实验室开发的C++库,提供了许多用于网络编程和并发处理的组件。它支持多种操作系统,包括Windows、Linux、Unix等。ACE的核心功能包括I/O复用、线程管理、定时器、信号处理、分布式...

    ACE 6.1.0 最新版

    ACE(Adaptive Communication Environment)是通信软件开发领域中的一款强大且灵活的开源框架,主要用于构建高性能、跨平台的网络应用程序。其最新版本为6.1.0,这标志着ACE在功能、稳定性和效率方面又有了进一步的...

Global site tag (gtag.js) - Google Analytics