现代C++(Modern C++)基本用法实践:八、线程支持
概述
在c++11之前,c++并未对线程编程提供直接的支持。在c++11之后,支持了线程管理、同步、条件变量等支持。
在其他的c++库中(例如UE的线程库)还增加了多任务模型的抽象。
用法举例
参考测试项目的modrenc_thread.cpp
文件
主要内容:
- 线程的创建
- 使用future&async进行异步操作
- 使用future&promise进行异步操作
- 有锁同步
- 无锁同步(原子操作)
#include "ModernCppTestHeader.h"
#include <thread>
#include <future>
#include <mutex>
namespace n_thread {
void increment(int n)
{
int i = 0;
while (++i < n)
{
LOG_VAR(i);
}
}
int accumulate(int n)
{
int r = 0;
for (size_t i = 0; i <= n; i++)
{
r += i;
}
LOG_VAR(r);
return r;
}
void accumulate_p(int n, std::promise<int> p)
{
int r = accumulate(n);
p.set_value(r);
}
std::mutex mtx;
void print_block_sync(int n, char c)
{
std::lock_guard<std::mutex> guard(mtx);
for (int i = 0; i < n; ++i) {
std::cout << c;
}
std::cout << '\n';
}
void print_block_async(int n, char c)
{
for (int i = 0; i < n; ++i) {
std::cout << c;
}
std::cout << '\n';
}
class Number
{
public:
Number(int v) :v(v) {}
int v;
};
void counter_async(Number* num, int n)
{
for (int i = 0; i < n; ++i) {
num->v++;
}
}
void counter_sync(std::atomic<int>* v, int n)
{
for (int i = 0; i < n; ++i) {
++(*v);
}
}
}
void thread_test()
{
LOG_FUNC();
LOG_TAG("线程创建");
{
std::thread t(n_thread::increment, 3);
t.join();
LOG("main thread goes on");
}
LOG_TAG("异步操作 使用 future&async");
{
std::future<int> f = std::async(n_thread::accumulate, 3);
LOG("等待异步计算");
int sum = f.get();
LOG_VAR(sum);
}
LOG_TAG("异步操作 使用 future&promise");
{
std::promise<int> p;
std::future<int> f = p.get_future();
std::thread t(n_thread::accumulate_p, 3, std::move(p));
LOG("等待异步计算");
int sum = f.get();
LOG_VAR(sum);
t.join();
}
LOG_TAG("有锁同步");
{
{
LOG("不同步");
std::thread t1(n_thread::print_block_async, 50, '#');
std::thread t2(n_thread::print_block_async, 50, '$');
t1.join();
t2.join();
}
{
LOG("同步");
std::thread t1(n_thread::print_block_sync, 50, '#');
std::thread t2(n_thread::print_block_sync, 50, '$');
t1.join();
t2.join();
}
}
LOG_TAG("无锁同步");
{
{
LOG("不同步, 有可能能正确计算,但是不稳定");
n_thread::Number n(0);
std::thread t1(n_thread::counter_async, &n, 100000);
std::thread t2(n_thread::counter_async, &n, 100000);
t1.join();
t2.join();
LOG_VAR(n.v);
}
{
LOG("同步");
std::atomic<int> v(0);
std::thread t1(n_thread::counter_sync, &v, 100000);
std::thread t2(n_thread::counter_sync, &v, 100000);
t1.join();
t2.join();
LOG_VAR(v);
}
}
}