我有点懵了,这么做会被卡住吗?线程玩的熟的帮忙看看。

weixi2864 2023-05-26 11:59:35

class Threadpool {

std::atomic_bool isPoolRunning_;

std::mutex taskQueMtx_; 

std::condition_variable notEmpty_;

...

}

 

析构函数

Threadpool::~Threadpool() {
isPoolRunning_ = false;
notEmpty_.notify_all();
std::unique_lock<std::mutex> lock(taskQueMtx_);
②exitCond_.wait(lock, [&]()-> bool { return threads_.size() == 0; });
}

 

线程函数

void Threadpool::threadFunc(int threadid)
{
auto lastTime = std::chrono::high_resolution_clock().now();

while(isPoolRunning_) {
③std::shared_ptr<Task> task;
//获取到任务,顺利释放锁
{
//要操作线程队列,先获取锁
④std::unique_lock<std::mutex> lock(taskQueMtx_);
std::cout << "tid=" << std::this_thread::get_id() << " 尝试获取任务" << std::endl;

......

}

 

有一套多线程的代码

我感觉退出的时候有互锁的可能性

主线程:到析构了①首先抢夺了锁

然后②陷入等待,但是锁没有释放

 

这时候如果子线程正好跑到③的位置

那是不是到④就互锁了?

 

百思不得其解,哪个高手帮着解释一下

关键是这玩意跑的很好,从来就没出过问题

...全文
162 7 打赏 收藏 转发到动态 举报
写回复
用AI写文章
7 条回复
切换为时间正序
请发表友善的回复…
发表回复
shuise 2023-06-10
  • 打赏
  • 举报
回复

std::cout << "tid=" << std::this_thread::get_id() << " 尝试获取任务" << std::endl; //这个有可能永远执行不到。不出错,因为是不重要的代码

赵4老师 2023-05-29
  • 打赏
  • 举报
回复

生产环境难道没有支持多线程的日志?

weixi2864 2023-05-29
  • 举报
回复
@赵4老师 没有专门的日志 不过这个服务整体都很稳定 感觉在生产环境跑的没问题
赵4老师 2023-05-30
  • 举报
回复
@weixi2864 判断生产环境跑的到底有没有问题,不应靠观察外部表现,而应靠检查内部足够详细的运行日志。
weixi2864 2023-06-02
  • 举报
回复
@赵4老师 嗯 你还是认为这段代码是有问题是吧?
于扶摇 2023-05-28
  • 打赏
  • 举报
回复

在析构函数中,线程pool在等待所有子线程退出时一直持有taskQueMtx_的锁,这会导致在子线程尝试获取任务时无法获取到锁,从而导致互锁的情况。为了避免这种情况,可以使用线程安全的等待所有子线程退出的方式,例如使用std:🧵:join_all()函数等待所有子线程退出。这样可以避免在等待过程中一直持有锁的情况,从而避免互锁问题。

weixi2864 2023-05-28
  • 举报
回复
@于扶摇 谢谢回复 我理解也是会卡住 关键这段代码一直在生产环境使用 从没有卡过。。。

65,005

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧