欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 健康 > 美食 > linux下的单例安全的线程池实现

linux下的单例安全的线程池实现

2025/6/28 6:22:39 来源:https://blog.csdn.net/2302_80190394/article/details/144531036  浏览:    关键词:linux下的单例安全的线程池实现

目录

一、引言

二、池化技术

三、代码实例

四、代码剖析

头文件和命名空间

结构体定义

模板类定义

私有成员函数(锁相关)

私有成员函数(队列相关)

公有成员函数(线程池操作)

单例模式实现

构造函数和析构函数

静态成员变量定义

静态成员变量初始化


一、引言

随着信息技术的飞速发展,服务器端应用程序面临着越来越高的并发处理需求。如何在保证系统稳定性和响应速度的同时,有效管理线程资源,成为了软件开发中的一个重要课题。单例模式的线程池,作为一种高效管理线程资源的设计模式,在Linux环境下得到了广泛的应用。本文将探讨如何在Linux系统中实现一个线程安全的单例线程池,确保全局只有一个线程池实例,并提供线程安全的数据访问和任务调度,从而为高并发、高可用性的服务端程序奠定坚实的基础。以下是单例安全线程池的实现要点和步骤。

二、池化技术

线程池包括:push到线程池的任务  2.多个准备处理任务的线程

存在一批线程所以一定:互斥 + 同步

因此需要使用锁和条件变量

三、代码实例

#pragma once#include <iostream>
#include <vector>
#include <string>
#include <queue>
#include <pthread.h>
#include <unistd.h>using namespace std;struct ThreadInfo
{pthread_t tid;string name;
};static const int defalutnum = 5;template<class T>
class ThreadPool
{
private:    //一些私有的关于锁的线程函数void Lock(){pthread_mutex_lock(&_mutex);} void Unlock(){pthread_mutex_unlock(&_mutex);}void Wait(){pthread_cond_wait(&_cond, &_mutex);}void Wake(){pthread_cond_signal(&_cond);}private:    bool IsQueueEmpty(){return _tasks.empty();}string GetThreadName(pthread_t tid)     //根据线程ID,获取线程名{for (aoto& info : _threads){if (info.tid == tid){return info.name;}}return "No such thread";    }public:void start()   //启动线程池{int num = _threads.size();for (int i = 0; i < num; i++){pthread_create(&_threads[i].tid, nullptr, HandlerTask, this);      //传this,让ThredHandldler函数变成静态(减少一个this参数)_threads[i].name = "Thread-" + to_string(i + 1);}}T Pop(){T t = _tasks.front();   //类型就是任务,取出队头任务_tasks.pop();return t;   }void Push(const T& t){Lock();_tasks.push(t);Wake();     //(没有任务之后,进行wait)push之后就可以唤醒线程Unlock();}/*当等待队列有数据时(即有至少一个线程正在等待这个条件变量):pthread_cond_signal 调用成功时,它会唤醒等待该条件变量的一个线程(在多个线程等待的情况下,具体唤醒哪一个线程取决于线程调度策略和实现)。返回值通常是 0,表示成功。当等待队列没有数据时(即没有线程正在等待这个条件变量):pthread_cond_signal 调用仍然会成功执行,但是因为没有线程在等待,所以实际上没有线程被唤醒。返回值仍然是 0,表示成功。*/static ThreadPool<T>* GetInstance()   //单例模式(一般静态成员会提供静态方法){if (_instance == nullptr)       //防止重复加锁{pthread_mutex_lock(&_lock);if (_instance == nullptr){std::cout << "log: singleton create done first!" << std::endl;_instance = new ThreadPool<T>();    //加锁申请}pthread_mutex_unlock(&_lock);}return _instance;}private:    //默认成员函数ThreadPool(){pthread_mutex_init(&_mutex, nullptr);pthread_cond_init(&_cond, nullptr);_threads.resize(defalutnum);   //默认创建5个线程}~ThreadPool()   {pthread_mutex_destroy(&_mutex);pthread_cond_destroy(&_cond);//全局锁和条件变量不需要销毁//STL自定义类型不用管// 内置类型不用管}  ThreadPool<T>& operator=(const ThreadPool<T>& t1) = delete;   //禁止拷贝构造ThreadPool(const ThreadPool<T>& t1) = delete;   //禁止拷贝构造private: vector<ThreadInfo> _threads;    //线程信息queue<T> _tasks;           //任务队列pthread_mutex_t _mutex;     //控制内部线程的互斥锁pthread_cond_t _cond;       //控制内部线程的条件变量static ThreadPool<T>* _instance;   //单例模式static pthread_mutex_t _lock;   //实例化单例时,进行保护的专门的锁
};template<class T>   //使用模板时,需要点名模板,点名类域时,需要加上模板参数(模板实例化之后才是类)
ThreadPool<T>* ThreadPool<T>::_instance = nullptr;   //单例模式template<class T>
pthread_mutex_t ThreadPool<T>::_lock = PTHREAD_MUTEX_INITIALIZER;   //实例化单例时,进行保护的专门的锁

需要注意的是

1.线程的执行方法只能有一个参数void * arg

但是由于是类内方法,所以会有一个隐藏的this指针。为了避免多出一个参数,我们选择将方法变成静态,这样手动去传入this指针。

2.这是懒汉模式的单例,为了创建单例时,每次都不需要申请锁,这样外部额外包裹了一层判断,这层判断可以避免已经申请好单例之后(指针不为空),直接避免进入if内容,从而直接返回。

3.在pop的时候没有加锁,那么调用pop时就得手动加锁。

四、代码剖析

头文件和命名空间

#pragma once
#include <iostream>
#include <vector>
#include <string>
#include <queue>
#include <pthread.h>
#include <unistd.h>
using namespace std;
  • #pragma once:防止头文件被多次包含。
  • 包含了必要的标准库头文件,如<iostream><vector><string><queue><pthread.h><unistd.h>
  • 使用了std命名空间

结构体定义

struct ThreadInfo {
    pthread_t tid;
    string name;
};

  • ThreadInfo结构体用于存储线程信息,包括线程ID (tid) 和线程名称 (name)。

模板类定义

template<class T>
class ThreadPool {
    // ...
};
 

私有成员函数(锁相关)

private:    //一些私有的关于锁的线程函数
void Lock() { pthread_mutex_lock(&_mutex); } 
void Unlock() { pthread_mutex_unlock(&_mutex); }
void Wait() { pthread_cond_wait(&_cond, &_mutex); }
void Wake() { pthread_cond_signal(&_cond); }
  • LockUnlock用于加锁和解锁互斥锁。
  • Wait用于等待条件变量。
  • Wake用于唤醒等待条件变量的线程。

私有成员函数(队列相关)

private:    
bool IsQueueEmpty() { return _tasks.empty(); }
string GetThreadName(pthread_t tid) {for (auto& info : _threads) {if (info.tid == tid) {return info.name;}}return "No such thread";    
}
  • IsQueueEmpty检查任务队列是否为空。
  • GetThreadName根据线程ID获取线程名称。

公有成员函数(线程池操作)

public:
void start() {int num = _threads.size();for (int i = 0; i < num; i++) {pthread_create(&_threads[i].tid, nullptr, HandlerTask, this);      //传this,让ThredHandldler函数变成静态(减少一个this参数)_threads[i].name = "Thread-" + to_string(i + 1);}
}
T Pop() {T t = _tasks.front();   //类型就是任务,取出队头任务_tasks.pop();return t;   
}
void Push(const T& t) {Lock();_tasks.push(t);Wake();     //(没有任务之后,进行wait)push之后就可以唤醒线程Unlock();
}
  • start启动线程池,创建指定数量的线程并初始化线程信息。
  • Pop从任务队列中取出一个任务。
  • Push将任务添加到任务队列,并唤醒等待的线程。

单例模式实现

static ThreadPool<T>* GetInstance() {if (_instance == nullptr) {       //防止重复加锁pthread_mutex_lock(&_lock);if (_instance == nullptr) {std::cout << "log: singleton create done first!" << std::endl;_instance = new ThreadPool<T>();    //加锁申请}pthread_mutex_unlock(&_lock);}return _instance;
}
  • GetInstance方法实现单例模式,确保全局只有一个线程池实例。

构造函数和析构函数

private:    //默认成员函数
ThreadPool() {pthread_mutex_init(&_mutex, nullptr);pthread_cond_init(&_cond, nullptr);_threads.resize(defalutnum);   //默认创建5个线程
}
~ThreadPool() {pthread_mutex_destroy(&_mutex);pthread_cond_destroy(&_cond);
}  
ThreadPool<T>& operator=(const ThreadPool<T>& t1) = delete;   //禁止拷贝构造
ThreadPool(const ThreadPool<T>& t1) = delete;   //禁止拷贝构造
  • 构造函数初始化互斥锁、条件变量,并设置默认线程数量。
  • 析构函数销毁互斥锁和条件变量。
  • 禁止拷贝构造和赋值操作。

静态成员变量定义

private:    //默认成员函数
vector<ThreadInfo> _threads;    //线程信息
queue<T> _tasks;           //任务队列
pthread_mutex_t _mutex;     //控制内部线程的互斥锁
pthread_cond_t _cond;       //控制内部线程的条件变量
static ThreadPool<T>* _instance;   //单例模式
static pthread_mutex_t _lock;   //实例化单例时,进行保护的专门的锁
  • _threads存储线程信息。
  • _tasks存储任务队列。
  • _mutex_cond分别用于同步访问任务队列。
  • _instance是单例模式的实例指针。
  • _lock用于保护单例实例的创建过程。

静态成员变量初始化

template<class T>   //使用模板时,需要点名模板,点名类域时,需要加上模板参数(模板实例化之后才是类)
ThreadPool<T>* ThreadPool<T>::_instance = nullptr;   //单例模式
template<class T>
pthread_mutex_t ThreadPool<T>::_lock = PTHREAD_MUTEX_INITIALIZER;   //实例化单例时,进行保护的专门的锁
  • 初始化静态成员变量。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词