0

0

c++中如何使用std::ref_c++传递引用给线程或绑定的方法【详解】

穿越時空

穿越時空

发布时间:2026-01-17 13:33:39

|

986人浏览过

|

来源于php中文网

原创

std::ref是包装左值引用的可拷贝对象,用于在std::thread或std::bind中传递引用语义;它不延长对象生命周期,要求被包装变量在整个调用期间有效,且形参必须声明为引用类型。

c++中如何使用std::ref_c++传递引用给线程或绑定的方法【详解】

std::ref 本质是包装引用的可拷贝对象

std::ref 不是“传递引用”本身,而是把一个左值引用包装成一个可拷贝、可移动的对象,从而绕过 std::threadstd::bind 默认的值语义拷贝行为。它内部持有原始对象的指针(或引用),解引用时才真正访问原变量。

常见错误是以为 std::ref(x) 让线程“拥有”了 x 的生命周期 —— 实际上它不延长生存期,若原变量在子线程执行前销毁,访问就是未定义行为。

  • 必须确保被 std::ref 包装的变量在整个绑定对象或线程运行期间持续有效
  • std::ref 只接受左值(即命名变量),不能用于字面量或临时对象:std::ref(42) 编译失败
  • 对应还有 std::cref,用于只读引用(返回 const T&

用 std::ref 传引用给 std::thread

默认情况下,std::thread 构造函数会对所有参数做拷贝(调用拷贝构造),即使你传的是引用类型,也会拷出一份副本。要让线程函数真正修改外部变量,必须显式用 std::ref 包装。

int value = 10;
auto func = [](int& v) { v *= 2; };
std::thread t(func, std::ref(value));
t.join();
// 此时 value == 20

注意:如果线程函数参数声明为 int(非引用),哪怕你传 std::ref(value),也会触发 int& → int隐式转换并拷贝值,失去引用效果。参数类型必须匹配。

立即学习C++免费学习笔记(深入)”;

Live PPT
Live PPT

一款AI智能化生成演示内容的在线工具。只需输入一句话、粘贴一段内容、或者导入文件,AI生成高质量PPT。

下载
  • 线程函数形参必须声明为引用类型(如 int&std::string&
  • 传参时用 std::ref(var),不是 &var(后者传指针,且无法通过类型推导自动转引用)
  • 避免在 lambda 捕获列表中用 [&] 试图“共享”变量 —— 捕获的是创建线程时的上下文,子线程可能已析构

std::ref 在 std::bind 中的必要性

std::bind 同样默认按值保存参数。即使你 bind 一个引用参数的函数,不加 std::ref,绑定对象里存的仍是副本。

void increment(int& x) { ++x; }
int a = 5;
auto bound = std::bind(increment, std::ref(a)); // ✅ 修改 a
bound();
// a == 6

auto bound_bad = std::bind(increment, a); // ❌ 拷贝 a,increment 修改的是副本
bound_bad();
// a 仍是 5

另一个典型场景是绑定成员函数时传引用参数:

struct Counter { int val = 0; void add(int& delta) { val += delta; } };
Counter c;
int step = 3;
std::bind(&Counter::add, &c, std::ref(step))(); // step 被引用传入
  • std::bindstd::ref 的处理是特化的:调用时自动解引用,还原为原始引用
  • 不要对 std::ref 做二次包装(如 std::ref(std::ref(x))),编译器通常拒绝或行为未定义
  • C++17 起推荐优先用 std::thread 直接构造 + lambda 捕获,比 std::bind 更直观;但遗留代码或需延迟绑定时仍需 std::ref

std::ref 的替代方案与边界情况

并非所有“传引用”需求都该用 std::ref。有些场景更安全或更自然:

  • 用指针(&x)+ 函数参数为 T*:明确所有权和空值风险,适合需要判空或可选修改的逻辑
  • std::shared_ptr 管理对象生命周期:当多个线程需共享并延长对象生存期时,比裸引用 + std::ref 更健壮
  • lambda 按引用捕获([&x]):仅适用于线程立即启动且父作用域生命周期可控的情况;不可用于存储到容器或延迟调用

std::ref 是个轻量包装器,不参与内存管理,也不做线程同步。如果你在线程间通过 std::ref 共享变量,务必自行加锁(如 std::mutex)或使用原子类型,否则数据竞争依然存在。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

317

2023.08.02

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

317

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

204

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号