目录
这篇博客我们主要基于前面的知识实现一个线程池(thread pool),我们想要的效果就是说提前把线程都创建好,等到有任务的时候直接往任务队列里放任务,线程直接去任务队列中取任务并完成即可,其中我们要引入日志的概念,就是把进程运行的信息以打印到屏幕或打印到文件的形式让我们看见。
其中有很多的语法细节等我们遇到了再说
bind的实际使用
如果我们想让类成员函数充当线程对象的回调函数,因为类成员函数第一个参数是this指针,与线程对象需要的函数的参数类型不同,此时我们就可以用函数绑定bind
我们需要知道的是bind的参数位置对应形参 ,placeholders::_1的这个1代表第一个实参,比如下面类内的两个函数
我们实际上线程对象想要的其实是这种类型的函数
通过bind就可以修改函数参数个数
上面的例子就是告诉我们:类的成员方法也可以成为另一个类的回调方法
日志如何实现
接下来就是引入日志,日志其实就是一条信息,其中包括这条日志的性质(正常,出问题,等等),日志打印的时间,那个文件的哪一行打印的这条日志,以及我们可以加入可变参数来自定义打印的形式,可变参数就是类似printf的参数
日志的性质我们可以使用枚举来实现,可以分为不同的等级
至于如何获取此时的时间我们可以用下面的接口
利用图一的函数可以获得一个时间戳,把这个时间戳放到图二的函数中可以得到一个结构体对象(如图三所示)的指针,通过这个结构体我们就可以获得我们想要的年月日时分秒,具体使用在下面代码中会有
所在的文件名和行数我们可以用下面的宏来获得
下面比较复杂的就是如何处理可变参数
man va_arg
通过这一系列接口我们就可以处理可变参数,下面我们先来简单的用一下
cpp#include <stdarg.h> void Test(int num, ...) // num表示后面还有的参数个数 { va_list arg; va_start(arg, num); while (num--) { int data = va_arg(arg, int); cout << data << ' '; } va_end(arg); } int main() { Test(5, 11, 22, 33, 44, 55); return 0; }
其实原理就是说参数会从左向右依次压入栈,通过第一个参数的地址和各个参数的类型我们就可以找到各个参数,这就是为什么printf中要传%d %c %s等等这些可以表示类型的东西了,printf有了第一个参数的地址,以及后面参数的类型就可以找到所有的参数了
难道我们也需要像printf中那样一点一点去解析吗?其实我们可以借助下面的接口
我们把参数都传进去就可以得到一个处理好的字符串
因为但凡用到日志的地方都需要传__LINE__和__FILE__,我们如何让它自动传这两个东西呢?可以用到宏,因为宏的本质就是一种替换为了让宏在不传可变参数的时候也能用,需要加##
并且为了让宏用起来更加安全,可以用do while的形式以代码块的形式去进行替换
我们既可以向屏幕上打印,也可以向文件中打印 ,并且为了打印的安全,我们可以给打印加锁
具体代码
cpp
//Main.cc
#include <iostream>
#include "Threadpool.hpp"
#include "Task.hpp"
#include <ctime>
#include <cstdlib>
int main()
{
srand(time(nullptr));
// Enablescreen();
EnableFile();
ThreadPool<Task> tp(5);
tp.InitThreadPool();
tp.Start();
int tasknum = 10;
while (tasknum)
{
int a = rand() % 10 + 1;
usleep(1234);
int b = rand() % 5 + 1;
Task t(a, b);
LOG(INFO, "main thread push task: %s", t.DebugToString().c_str());
tp.Enqueue(t);
sleep(1);
tasknum--;
}
tp.Wait();
return 0;
}
cpp
//ThreadPool.hpp
#pragma once
#include <iostream>
#include <vector>
#include <queue>
#include <string>
#include <memory>
#include <pthread.h>
#include <unistd.h>
#include "MyThread.hpp"
#include"Task.hpp"
#include"Log.hpp"
const static int gdefaultthreadnum = 3;
template <class T>
class ThreadPool
{
public:
ThreadPool(int threadnum = gdefaultthreadnum)
: _threadnum(threadnum), _waitnum(0), _isrunning(false)
{
pthread_mutex_init(&_mutex, nullptr);
pthread_cond_init(&_cond, nullptr);
LOG(INFO, "ThreadPool Construct()");
}
void HandlerTask(std::string &name)
{
LOG(INFO, "%s is running...", name.c_str());
while (true)
{
pthread_mutex_lock(&_mutex);
while (_task_queue.empty() && _isrunning)
{
_waitnum++;
pthread_cond_wait(&_cond, &_mutex);
_waitnum--;
}
if (_task_queue.empty() && !_isrunning)
{
pthread_mutex_unlock(&_mutex);
break;
}
T t = _task_queue.front();
_task_queue.pop();
pthread_mutex_unlock(&_mutex);
LOG(DEBUG, "%s get a task", name.c_str());
t();
LOG(DEBUG, "%s handler a task, result is: %s", name.c_str(), t.ResultToString().c_str());
}
}
void InitThreadPool()
{
for (int i = 0; i < _threadnum; i++)
{
std::string name = "thread-" + std::to_string(i + 1);
_threads.emplace_back(std::bind(&ThreadPool::HandlerTask, this, std::placeholders::_1), name);
LOG(INFO, "init thread %s done", name.c_str());
}
_isrunning = true;
}
void Start()
{
for (auto &thread : _threads)
{
thread.start();
}
}
void Stop()
{
pthread_mutex_lock(&_mutex);
_isrunning = false;
pthread_cond_broadcast(&_cond);
pthread_mutex_unlock(&_mutex);
}
bool Enqueue(const T &t)
{
bool ret = false;
pthread_mutex_lock(&_mutex);
if (_isrunning)
{
_task_queue.push(t);
if (_waitnum > 0)
{
pthread_cond_signal(&_cond);
}
LOG(DEBUG, "enqueue task success");
ret = true;
}
pthread_mutex_unlock(&_mutex);
return ret;
}
void Wait()
{
for (auto &thread : _threads)
{
thread.join();
LOG(INFO, "%s is quit...", thread.name().c_str());
}
}
~ThreadPool()
{
pthread_mutex_destroy(&_mutex);
pthread_cond_destroy(&_cond);
}
private:
int _threadnum;
std::vector<Thread<T>> _threads;
std::queue<T> _task_queue;
pthread_mutex_t _mutex;
pthread_cond_t _cond;
int _waitnum;
bool _isrunning;
};
cpp
//MyThread.hpp
#pragma once
#include <iostream>
#include <string>
#include <functional>
#include <pthread.h>
using namespace std;
using fun_t = function<void(string&)>;
template <class T>
class Thread
{
private:
void excute(string& name)
{
_func(name);
}
public:
Thread(fun_t func, const string&name="noname")
: _func(func), _name(name), _stop(true) {}
static void *threadrun(void *args)//如果不是静态,会有this指针
{
Thread<T> *ptr = reinterpret_cast<Thread<T> *>(args);
ptr->excute(ptr->_name);
return nullptr;
}
bool start()
{
int n = pthread_create(&_id, nullptr, threadrun, this);//把this当参数传过去
if (n == 0)
{
_stop = false;
return true;
}
else
{
return false;
}
}
void join()
{
if (!_stop)
{
pthread_join(_id, nullptr);
}
}
void detach()
{
if (!_stop)
{
pthread_detach(_id);
}
}
void stop()
{
_stop = true;
}
string name()
{
return _name;
}
private:
pthread_t _id;
string _name;
bool _stop;
fun_t _func;
};
cpp
//Log.hpp
#pragma once
#include <iostream>
#include <string>
#include<fstream>
#include <ctime>
#include <cstdarg>
#include <sys/types.h>
#include <unistd.h>
#include <pthread.h>
bool gIsSave = false;
const char *logname = "log.txt";
enum Level
{
DEBUG,
INFO,
WARNING,
ERROR,
FATAL
};
std::string LevelToString(int level)
{
switch (level)
{
case DEBUG:
return "Debug";
case INFO:
return "Info";
case WARNING:
return "Warning";
case ERROR:
return "Error";
case FATAL:
return "Fatal";
default:
return "Unknown";
}
}
std::string GetTimeString()
{
time_t curr_time = time(nullptr); // 获取当前时间戳
struct tm *format_time = localtime(&curr_time);
if (format_time == nullptr)
return "None_time";
char time_buffer[1024];
snprintf(time_buffer, sizeof(time_buffer), "%d-%d-%d %d:%d:%d",
format_time->tm_year + 1900,
format_time->tm_mon + 1,
format_time->tm_mday,
format_time->tm_hour,
format_time->tm_min,
format_time->tm_sec);
return time_buffer;
}
pthread_mutex_t glock = PTHREAD_MUTEX_INITIALIZER;
void SaveMessage(const char *filename, std::string &message)
{
std::ofstream out(filename, std::ios::app);
if (!out.is_open())
{
return;
}
out << message;
out.close();
}
void LogMessage(std::string filename, int line, bool issave, int level, const char *format, ...)
{
std::string levelstr = LevelToString(level);
std::string timestr = GetTimeString();
pid_t selfid = getpid();
char buffer[1024];
va_list arg;
va_start(arg, format); // arg和第一个参数
vsnprintf(buffer, sizeof(buffer), format, arg);
va_end(arg);
std::string message = "[" + timestr + "]" + "[" + levelstr + "]" +
"[" + std::to_string(selfid) + "]" +
"[" + filename + "]" + "[" + std::to_string(line) + "] " + buffer + "\n";
pthread_mutex_lock(&glock);
if (!issave) // 如果不用保存
{
std::cout << message;
}
else
{
SaveMessage(logname, message);
}
pthread_mutex_unlock(&glock);
}
#define LOG(level,format,...) do{LogMessage(__FILE__,__LINE__,gIsSave,level,format,##__VA_ARGS__);}while(0)
#define EnableFile() do{gIsSave=true;}while(0)
#define Enablescreen() do{gIsSave=false;}while(0)
cpp
//Task.hpp
#pragma once
#include <iostream>
#include <string>
#include <functional>
class Task
{
public:
Task() {}
Task(int a, int b) : _a(a), _b(b), _result(0)
{
}
void Excute()
{
_result = _a + _b;
}
std::string ResultToString()
{
return std::to_string(_a) + "+" + std::to_string(_b) + "=" + std::to_string(_result);
}
std::string DebugToString()
{
return std::to_string(_a) + "+" + std::to_string(_b) + "=?";
}
void operator()()
{
Excute();
}
private:
int _a;
int _b;
int _result;
};