背景
- 管理器用于管理多个任务
- 接入了管理器的模块,会根据自己要做的事情,来生成不同的任务
- 而这些任务的生成,是由接入了管理器的模块,通过一些重要的数据(比如目标数据包的编号,以及目标所在的位置,要求管理器执行的任务类型),来让管理器生成的不同的任务,并添加到任务队列中
- 目前的设计的框架是支持多种任务的,但目前仅实现了安装、卸载、解压三种任务
- 通俗来讲,接入了管理器的模块,告诉管理器要处理的对象的编号,要处理的对象的所在位置,以及这个任务的任务类型,由管理器具体去执行其他操作比如生成一个任务、执行任务、管理任务等等,不管执行成功失败,再由管理器把结果通知到接入了管理器的模块
设计与实现
状态通知
- 为了让管理器能做到将任务状态通知给接入模块,设计如下:
- 让接入模块继承了管理器的委托模块,这样从继承的角度来看,接入了管理器的模块自己就是一个委托模块,于是管理器可以在适当的时机委托接入模块去做一些事情
- 比如,当安装功能完成时,管理器自己是知道的,但是接入了管理器的模块是不知道已经完成了,这时管理器可以委托接入模块去做一些安装成功后该做的事情,也就是相当于通知接入模块某件事达成了,去接管后续
1 2 3 4 5 |
class AppCenterAppExec : public appcenter_iappexec , public ac_help::appcenter_helper_file::delegate { // ... } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
class appcenter_helper_file : public appcenter_ihelper , public file_helper::checker , public installer_helper::installer , public uninstaller_helper ::uninstaller , public unzip_helper ::unziper { public: class delegate { friend class appcenter_helper_file; virtual void notify(install_type type, const char* appid) = 0; virtual void set_history_data(const char* appid, const wchar_t* data) = 0; virtual std::wstring get_unist_goal(const char* appid) = 0; }; // ... }; |
单例管理器
- 由于接入模块可能在不同的线程中对管理器添加任务,而我们希望由管理器接管处理所有的任务,这种情况下,就需要接入模块的不同线程操作的都是同一个管理器对象
- 于是,我把管理器设计成了单例的,只在第一次进行初始化,后面通过接口去获取到的管理器,都会是同一个
- 可以看到这个类的拷贝构造和拷贝赋值操作符都是
delete
了,并且构造函数是private
的 - 这样就只能通过给出的
public
接口get_helper
来获取这个管理器的实例了
- 可以看到这个类的拷贝构造和拷贝赋值操作符都是
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 |
class appcenter_helper_file : public appcenter_ihelper , public file_helper::checker , public installer_helper::installer , public uninstaller_helper ::uninstaller , public unzip_helper ::unziper { public: class delegate { friend class appcenter_helper_file; virtual void notify(install_type type, const char* appid) = 0; virtual void set_history_data(const char* appid, const wchar_t* data) = 0; virtual std::wstring get_unist_goal(const char* appid) = 0; }; static appcenter_helper_file* get_helper(delegate* app); appcenter_helper_file(const appcenter_helper_file&) = delete; appcenter_helper_file& operator=(const appcenter_helper_file&) = delete; ~appcenter_helper_file() override; // from appcenter_ihelper void start(start_type type) override; void add_task(const char* appid, const char* app_path, start_type type); bool get_first_status(); void update_first_status(bool status); static void th_cls(); private: explicit appcenter_helper_file(delegate* app); // ... }; |
- 可以看到
get_helper
是一个static
的函数- 并且通过指定了一些内存序来获取或创建这个管理器
- 第一次指定了
memory_order_acquire
的意思,就是确保如果在另一个线程里面已经完成了对这个管理器的初始化(new appcenter_helper_file(app)
)之后的所有写入操作对当前线程是可见的 - 第二次用
memory_order_relaxed
是因为我们已经在一个锁的保护下,所以不需要额外的内存顺序保证。换句话说,当已经获取到了锁的保护了,这样读的时候,是不会有其他线程在写的,所以用最宽松的内存序也是可以了 - 第三次用
memory_order_release
是因为,我们正在写,我们需要确保所有的写操作在存储到m_helper
之前都完成,让任何看到m_helper
不为空的线程看到的m_helper
是一个完成初始化的对象
1 2 3 4 5 6 7 8 9 10 11 12 |
appcenter_helper_file* appcenter_helper_file::get_helper(delegate* app) { appcenter_helper_file* temp = m_helper.load(std::memory_order_acquire); if (temp == nullptr) { std::lock_guard<std::mutex> lock(m_mutex); temp = m_helper.load(std::memory_order_relaxed); if (temp == nullptr) { temp = new appcenter_helper_file(app); m_helper.store(temp, std::memory_order_release); } } return temp; } |
添加任务
- 从上面可以看到,管理器提供了一个
add_task
接口,这个接口需要3个参数,然后,根据任务类型,任务将被添加到不同的任务列表里- 当然,在添加的时候,使用了互斥量进行了同步
- 可以在下面代码中看到,安装和卸载被添加到了同一个任务列表中了,而解压被添加到了另外的任务列表中了,这是根据这些活动占CPU的不同,在设计上对它们进行了分类
- 设计为:低CPU任务由单独的线程处理,高CPU任务也由单独的线程处理,这两个线程并发运行。但是,这两个线程同时只能运行一个任务
- 当然,当前框架稍作修改,就可以改为每类任务能同时运行多个,之所以按上述设计,是业务需求
- 每当一个任务被添加到对应的任务队列,就使用条件变量去唤醒对应的线程
- 并且,管理器记录当当前正在处理的低类和高类任务,所以在添加任务时,对相同的任务进行了过滤
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 |
void appcenter_helper_file::add_task(const char* appid, const char* app_path, start_type type) { switch (type) { case ac_help::start_type::em_st_install: { bool flag = false; { std::unique_lock<std::mutex> _lck(m_mutex_thread_low); auto tmp_task = m_ist_cur.find(appid); if (tmp_task != m_ist_cur.end()) { flag = true; } } if (flag) { install_status(install_type::em_it_other_err_same_install_task_installing, appid); } else { { std::unique_lock<std::mutex> _lck(m_mutex_thread_low); m_low_list.push_back(new install_task(appid, app_path)); m_cv_low.notify_one(); } } break; } case ac_help::start_type::em_st_uninstall: { bool flag = false; { std::unique_lock<std::mutex> _lck(m_mutex_thread_low); auto tmp_task = m_ist_cur.find(appid); if (tmp_task != m_ist_cur.end()) { flag = true; } } if (flag) { install_status(install_type::em_it_other_err_same_install_task_installing, appid); } else { { std::unique_lock<std::mutex> _lck(m_mutex_thread_low); m_low_list.push_back(new uninstall_task(appid, app_path)); m_cv_low.notify_one(); } } break; } case ac_help::start_type::em_st_start: break; case ac_help::start_type::em_st_stop: break; case ac_help::start_type::em_st_unzip: { bool flag = false; { std::unique_lock<std::mutex> _lck(m_mutex_thread_high); auto tmp_task = m_unzip_cur.find(appid); if (tmp_task != m_unzip_cur.end()) { flag = true; } } if (flag) { unzip_status(install_type::em_it_other_err_same_unzip_task_installing, appid); } else { { std::unique_lock<std::mutex> _lck(m_mutex_thread_high); m_high_list.push_back(new unzip_task(appid, app_path)); m_cv_high.notify_one(); } } break; } default: break; } if (!m_first) { m_first = true; } } |
低CPU线程
- 可以看到,这个线程处理低CPU任务,由于每个任务标记了任务类型,所以在这里,可以根据任务的类型对不同的任务进行了处理
- 目前的实现可以看到仅仅把安装卸载归为低CPU任务,并在这里进行了处理
- 至于安装卸载的具体事项,见其他文章
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 |
void appcenter_helper_file::install_proc() { while (true) { low_task_helper* current_task = nullptr; task_group type(task_group::em_tg_low_group); { std::unique_lock<std::mutex> lock(m_mutex_thread_low); m_cv_low.wait(lock, [this]() { return m_stop_sig || (!m_low_list.empty() && !m_is_low); }); if (m_stop_sig) { break; } if (m_low_list.empty()) { continue; } current_task = m_low_list.front(); if (!current_task) { install_status(install_type::em_it_other_err_task_err, ""); continue; } type = current_task->get_task_group(); if (type == task_group::em_tg_low_group && m_is_low) { continue; } else { m_is_low = true; } m_low_list.pop_front(); } task_type task_type = current_task->get_task_type(); std::string appid = current_task->get_appid(); std::string app_path = current_task->get_app_path(); delete current_task; current_task = nullptr; if (task_type == task_type::em_tk_insall) { start_with_type_install(appid.c_str(), app_path.c_str()); } else if (task_type == task_type::em_tk_uninstall) { start_with_type_uninstall(appid.c_str()); } else {} } } |
高CPU线程
- 可以看到,这个线程处理高CPU任务
- 目前仅把解压归为高CPU任务
- 同样,解压具体实现,见其他文章
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 |
void appcenter_helper_file::unzip_proc() { while (true) { high_task_helper* current_task = nullptr; task_group type(task_group::em_tg_high_group); { std::unique_lock<std::mutex> lock(m_mutex_thread_high); m_cv_high.wait(lock, [this]() { return m_stop_sig || (!m_high_list.empty() && !m_is_high); }); if (m_stop_sig) { break; } if (m_high_list.empty()) { continue; } current_task = m_high_list.front(); if (!current_task) { unzip_status(install_type::em_it_other_err_unzip_failed, ""); continue; } type = current_task->get_task_group(); if (type == task_group::em_tg_high_group && m_is_high) { continue; } else { m_is_high = true; } m_high_list.pop_front(); } task_type task_type = current_task->get_task_type(); std::string appid = current_task->get_appid(); std::string app_path = current_task->get_app_path(); delete current_task; current_task = nullptr; if (task_type == task_type::em_tk_unzip) { start_with_type_unzip(appid.c_str(), app_path.c_str()); } else {} } } |
线程初始化
- 由于需求相关,所以在设计上并没有采用线程池的做法,而是直接创建了两条线程用来处理不同的任务
1 2 3 4 5 6 7 8 9 10 11 12 13 |
appcenter_helper_file::appcenter_helper_file(delegate* app) : m_app(app) , m_first(false) , m_is_low(false) , m_is_high(false) , m_stop_sig(false) , m_low_group(task_group::em_tg_default_group) , m_high_group(task_group::em_tg_default_group) , m_low_goal_path(nullptr) , m_high_goal_path(nullptr) { m_th_low = std::make_unique<std::thread>(&appcenter_helper_file::install_proc, this); m_th_high = std::make_unique<std::thread>(&appcenter_helper_file::unzip_proc, this); } |
管理器的清理
- 由于管理器是被其他模块接入的,所以管理器的清理工作,肯定是在其他模块的清理时机去做才比较合理,如下:
- 管理器提供了
static
方法th_cls
,给外部接口去调用,这个函数的功能是获取管理器类的指针后,用delete
去释放 - 从而,会调到管理器的析构函数
- 而真正的资源清理,是在管理器的析构里面去做的
- 管理器提供了
1 2 3 4 5 6 7 8 9 10 |
void AppCenterAppExec::uninit() { ac_help::appcenter_helper_file* helper = ac_help::appcenter_helper_file::get_helper(this); if (helper) { helper->th_cls(); } delete this; } |
1 2 3 4 5 6 7 |
void appcenter_helper_file::th_cls() { auto helper = appcenter_helper_file::m_helper.load(); if (helper) { delete helper; appcenter_helper_file::m_helper.store(nullptr); } } |
管理器析构
- 可以看到,先修改了一个原子类型的变量,置为
true
- 然后调用了
notify_one
来对应的线程去检测条件- 如果线程此时在休眠,那它会被唤醒,检查谓词后退出
- 如果现场正常运行,已经是被唤醒了,那么
notify_one
不会产生影响,这时候,就需要正常执行任务的线程在某个状态时做某些处理,从而及时结束任务
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
appcenter_helper_file::~appcenter_helper_file() { m_stop_sig = true; m_cv_low.notify_one(); m_cv_high.notify_one(); if (m_th_low && m_th_low->joinable()) { m_th_low->join(); m_th_low.reset(); } if (m_th_high && m_th_high->joinable()) { m_th_high->join(); m_th_high.reset(); } for (low_task_helper* task : m_low_list) { install_status(install_type::em_it_other_err_install_cancel, task->get_appid()); delete task; } m_low_list.clear(); for (high_task_helper* task : m_high_list) { install_status(install_type::em_it_other_err_install_cancel, task->get_appid()); delete task; } m_high_list.clear(); } |
- 然后,判刑线程是否可连接,如果可连接,调用
join
等待线程退出,这将导致当前线程被阻塞,知道这个线程完成了任务 - 在两个线程都退出后,检查任务队列里面是否还存在任务,如果存在,要用
delete
删掉任务,因为这些任务在添加的时候是被new
出来的
线程的状态恢复
- 当一个任务完成后,调用之前介绍的委托接口,来通知任务的完成状态
- 通知完后,上锁,将当前保存的任务设置为空,将线程运行标志清空(这个点是否可以优化),将当前线程对应的任务归类标志设置默认
- 调用
notify_one
唤醒线程,让他接着处理其他任务
相关优化点
清理线程时,任务正常进行
- 清理线程时,任务正在进行,那可以让线程在执行任务期间,检查某个状态,如果发现状态被设置了,那么就取消任务的后续执行
线程运行标志
- 只有一个线程的情况下,只要这个线程被唤醒了,那么再调用
notify_one
通知这个线程处理任务,是不会产生影响的,或者是说没有任何效果 - 所以上述设计中,线程运行标志似乎有点多余
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ 深度探索C++对象模型一02/09
- ♥ Pybind11记述:一07/04
- ♥ C++20_第二篇03/21
- ♥ Effective C++_第一篇01/10
- ♥ Boost 程序库完全开发指南:容器算法数学文件08/24
- ♥ C++_volatile10/08