std::thread 提供类型安全的线程管理,支持函数、lambda 创建线程,需用 join 或 detach 避免终止,推荐 RAII 守护和 std::ref 传引用。

在C++11及以后的标准中,std::thread 提供了简洁、类型安全的方式来创建和管理线程。相比传统的POSIX线程(pthread),它更易于使用,并能与现代C++特性如lambda表达式、函数对象等无缝结合。
创建线程
使用 std::thread 创建线程非常直接:只需将一个可调用对象(函数、lambda、函数对象)传递给其构造函数即可启动新线程。
示例:通过普通函数创建线程
void hello() {
std::cout
}
int main() {
std::thread t(hello);
t.join(); // 等待线程结束
return 0;
}
示例:使用lambda表达式
int main() {
std::thread t([]() {
std::cout
});
t.join();
return 0;
}
注意:传递参数给线程函数需按值或显式使用 std::ref 包装引用
立即学习“C++免费学习笔记(深入)”;
void print_value(int& x) {
x = 42;
}
int main() {
int val = 0;
std::thread t(print_value, std::ref(val)); // 必须用 std::ref
t.join();
std::cout
return 0;
}
线程的等待与分离
每个 std::thread 对象必须在销毁前决定是等待其完成还是将其分离,否则程序会调用 std::terminate() 终止。
- join():主线程阻塞等待子线程执行完毕。调用后,线程对象不再关联任何执行线程。
- detach():将线程设置为后台运行,不再与 std::thread 对象关联。此后无法再对该线程进行控制或等待。
常见做法:
- 若需要获取结果或确保线程完成,使用 join()
- 若线程执行长期任务且无需同步,可 detach(),但要小心资源管理和生命周期问题
线程管理技巧
直接使用裸的 std::thread 可能导致异常安全问题。例如,在创建多个线程时,若中间抛出异常,前面已创建的线程可能未被正确回收。
推荐做法:使用 RAII 技术封装线程管理。
简单线程守卫示例
class ThreadGuard {
std::thread t;
public:
explicit ThreadGuard(std::thread t_) : t(std::move(t_)) {}
~ThreadGuard() {
if (t.joinable()) t.join();
}
ThreadGuard(const ThreadGuard&) = delete;
ThreadGuard& operator=(const ThreadGuard&) = delete;
ThreadGuard(ThreadGuard&&) = default;
};
这样即使发生异常,析构函数也会自动调用 join(),避免资源泄漏。
常见注意事项
- 确保共享数据的线程安全,必要时配合 std::mutex 使用
- 不要忘记调用 join() 或 detach(),否则程序终止
- 传递参数时注意默认是按值拷贝,修改局部变量需使用引用包装器
- 避免 detach 的线程访问已销毁的局部变量或对象
基本上就这些。std::thread 让多线程编程变得直观,只要注意生命周期和同步问题,就能写出稳定高效的并发代码。











