答案:C++中std::thread默认按值传递参数,引用需用std::ref包装,成员函数需传对象指针,注意避免悬空引用。

在C++中使用std::thread创建线程时,传递参数是一个常见需求。正确理解参数传递方式可以避免潜在的错误,比如数据竞争或悬空引用。
默认按值传递参数
当你将参数传入std::thread构造函数时,默认是按值拷贝到线程内部的。这意味着即使你传的是变量名,实际在线程函数中接收到的是副本。
例如:
#include#include void func(int x, std::string str) { std::cout << "x = " << x << ", str = " << str << "\n"; } int main() { int a = 42; std::string s = "hello"; std::thread t(func, a, s); // 按值传递 t.join(); return 0; }
这里a和s都会被复制一份给线程函数func。
立即学习“C++免费学习笔记(深入)”;
通过引用传递参数
如果你想在线程中修改外部变量,必须显式使用std::ref或std::cref包装引用。
示例:
#include#include void increment(int& x) { ++x; } int main() { int value = 10; std::thread t(increment, std::ref(value)); // 正确传递引用 t.join(); std::cout << "value after thread: " << value << "\n"; // 输出 11 return 0; }
如果不加std::ref,increment接收的将是int的副本,无法修改原变量。
传递类成员函数作为线程函数
调用对象的成员函数时,第一个参数是对象指针或引用,后续才是成员函数的参数。
示例:
struct Worker {
void run(int id) {
std::cout << "Worker " << id << " is running\n";
}
};
Worker w;
std::thread t(&Worker::run, &w, 100); // &w 是 this 指针,100 是 run 的参数
t.join();
这里&Worker::run是成员函数指针,&w提供对象上下文,100是id参数。
注意生命周期问题
确保线程执行期间,所引用的对象仍然有效。特别是使用引用或指针时,避免出现悬空引用。
常见错误:
void bad_example() {
std::string data = "temporary";
std::thread t([](const std::string& s) {
// 可能访问已销毁的 data
std::cout << s << "\n";
}, std::ref(data));
t.detach(); // 线程可能还在运行
} // data 在此处析构
这种情况下,如果线程还未执行完,data已被销毁,会导致未定义行为。
基本上就这些。关键是记住:参数默认拷贝,引用需用std::ref,成员函数要传对象,生命周期要管好。










