C++实现线程池
Posted 小懵白生活小趣谈
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++实现线程池相关的知识,希望对你有一定的参考价值。
在上一篇文章中,小懵白就给大家简要地讲解了线程池的原理。
今天呢,小懵白就给大家继续讲解C++如何实现封装线程池类。
第一步
简要代码如下:
bool Init_ThreadPool(){
for(int i=0;i<max_thread;i++) //max_thread为初始化线程最大化量
pool.push_back(std::thread(&ThreadPool::runtask,this));
return true;
}
void runtask(){
while(!stop){
std::unique_lock<std::mutex>lk(_mutex);
//当任务队列为空或者stop为假时,阻塞当前线程,直到条件变量唤醒
condition.wait(lk,[this]{return (!task.empty()||stop);});
/**********************************************
std::move是将对象的状态或者所有权从一个对象转移到另一个对象,
只是转移,没有内存的搬迁或者内存拷贝。
***********************************************/
Task ta=move(task.front());// 取一个 task
task.pop(); //从队列移除正在执行的任务
lk.unlock();
ta(); //执行task任务
condition.notify_one(); //通知等待一个线程
}
//此处补充销毁线程代码
}
using Task = std::function<void()>;
接着就是实现生产者了:
bool Add_task(const Task&t){
if(task.size()==max_queue)Warn_LOG("添加任务,任务队列已满,准备阻塞");
if(thread_run==max_thread)Warn_LOG("添加任务,但线程已经分配完");
std::unique_lock<std::mutex>lk(_mutex);
while(task.size()==max_queue||stop){ //要是任务数量到了最大,就等待处理完再添加
condition.wait(lk);
}
if(stop)return false;
task.push(t);//给队列中添加任务
task_numble=task.size();
std::cout<<"添加成功。"<<std::endl<<std::endl;
condition.notify_one(); //通知等待一个线程
return true;
}
为此,到这里的时候,生产者和消费者的线程池初步构建完成了
第二步:解说
在生产者pool容器中,每一个线程都是绑定了runtask函数,也就是说每一个线程都是在执行runtask函数。
第三步:封装
实现完上面的功能后,接下来就是对功能进行封装了。代码如下:
class ThreadPool{
using Task = std::function<void()>;
public:
ThreadPool();
~ThreadPool();
bool Init_ThreadPool(); //初始化线程
bool Add_task(const Task&t); //添加任务
void End_threadpool(); //销毁线程
int thread_num; //当前的线程数量
int task_numble; //任务队列
protected:
void runtask();
void Expand_thread();
bool Destroy_thread();
int id;
private:
int max_thread;//初始化线程数量
int max_queue;//初始化 任务队列数量
std::vector<std::thread> pool;// 线程池、任务队列
std::queue<Task>task ;
std::mutex _mutex ;//互斥锁条件变量
std::condition_variable condition;
bool stop; //停止标志位
int key; //目前正在执行任务的线程数量
};
ThreadPool::~ThreadPool(){
End_threadpool();
}
void ThreadPool::End_threadpool(){
stop=true; //停止读取任务
while(thread_run){}//等待线程还没有执行完的任务
for(int i =0 ;i<pool.size();i++) //销毁线程
pool[i].join() ;
std::queue<Task> empty;
swap(empty,task);//清空任务队列
pool.clear(); //清空线程池
}
至此,线程池类封装就实现了,其余功能函数在这就不多讲了,有兴趣的同学可自行完成。
END
以上是关于C++实现线程池的主要内容,如果未能解决你的问题,请参考以下文章