目录
1、需求
需求:
有三条线程,可以分别输出A、B、C字符,现想办法控制输出顺序,使得顺序为ABCABC......ABC[共10组],并且每条线程只能启动一次,不能多次启动。
2、分析
分析:
要在标准C++中使用多线程,可以使用类 thread,3条线程,我们可以直白一些,直接实现3个线程函数,在函数里面进行循环输出对应字符,每个循环控制10次即可;
同时,要想控制输出顺序的话,最开始想到的方式是直接加一个标志位来控制,也是极好的,也基本上能实现ABC的输出效果;
通过一个标志位来处理的方式虽然可以实现效果,但是还不够安全,极有可能有多条线程同时操作一个全局变量,导致资源争夺问题,为了保证安全,可以在此基础上加上对应的锁来处理同步问题,比如加上互斥锁 mutex,就可以有效的防止资源争夺问题。
既然已经使用了mutex锁了,能不能在此基础上再优化以下,配合守卫锁 lock_guard 或者唯一锁 unique_lock 来解决可能导致的异常死锁问题。
最后,使用条件变量和互斥锁+唯一锁来解决这个需求即可。
----------------------------------------------------------
前面已经有2个帖子进行了代码的实现,但是都过于复杂,最后优化以下,得到更好的代码展示效果。
如果对前面的内容还不是很清楚的,可以查看一下前面的2个帖子,传送门如下:
<一>、C++实现多线程的同步处理:控制ABC的输出顺序,输出10组,mutex+condition_variable-优快云博客
<二>、C++实现多线程的同步处理:控制ABC的输出顺序,输出10组,mutex+condition_variable-优快云博客
---------------------------------------------------------
3、代码
直接放代码优化好的代码,大大的减少了函数的使用,同一个函数进行了复用,只需要改编成参数即可。
#include <iostream>
#include <mutex>
#include <thread>
#include <condition_variable>
using namespace std;
mutex g_mutex;
condition_variable g_conv; // 条件变量
int g_cnt = 0; // 计数器,初始化为0
void outABC(char c, int val)
{
for (int i = 0; i < 10; ++i) {
unique_lock<mutex> ul(g_mutex);
g_conv.wait(ul, [=] {return g_cnt % 3 == val; });
cout << c;
++g_cnt;
g_conv.notify_all();
}
}
int main()
{
// 需要理解的是同一个线程函数虽然起了3次,但是参数不一样,效果也不一样
thread tA(&outABC, 'A', 0);
thread tB(&outABC, 'B', 1);
thread tC(&outABC, 'C', 2);
tA.detach();
tB.detach();
tC.join();
return 0;
}