std::ref和std::cref的使用和原理分析

目录

1.用法

2.std::reference_wrapper介绍

3.std::ref原理分析

4.std::cref原理分析

5.总结


1.用法

        它的定义如下:

        std::ref:用于包装按引用传递的值。

        std::cref:用户包装按const引用传递的值。

        C++本身就有引用(&),那为什么C++11又引入了std::ref(或者std::cref)呢?

        这主要是考虑函数式编程(如std::bind或std::thread)在使用时,是对参数直接拷贝,而不是引用。这一点在讲解std::bind是也说的很清楚,bind函数的所有实参(含第1个实参)都是按值传递的。如果有不清楚的地方可参考其定义或下面的博客:

C++中的std::bind深入剖析-CSDN博客

        std::ref 和 std::cref 只是尝试模拟引用传递,并不能真正变成引用,在非模板情况下,std::ref根本没法实现引用传递,只有模板自动推到类型时,ref能包装类型reference_wrapper来代替原本会被识别的值类型,而reference_wrapper能隐式转换为被引用的值的引用类型,但是并不能被用作 & 类型。

        下面举个例子来说明一下它的用法:

#include <functional>
#include <iostream>void f(int& n1, int& n2, const int& n3)
{std::cout << "函数中: " << n1 << ' ' << n2 << ' ' << n3 << '\n';++n1; // 增加存储于函数对象的 n1 副本++n2; // 增加 main() 的 n2// ++n3; // 编译错误
}int main()
{int n1 = 1, n2 = 2, n3 = 3;std::function<void()> bound_f = std::bind(f, n1, std::ref(n2), std::cref(n3));n1 = 10;n2 = 11;n3 = 12;std::cout << "函数前: " << n1 << ' ' << n2 << ' ' << n3 << '\n';bound_f();std::cout << "函数后: " << n1 << ' ' << n2 << ' ' << n3 << '\n';
}

输出:

函数前: 10 11 12
函数中: 1 11 12
函数后: 10 12 12

        从上面的例子中可以看到,执行完f,n1的值仍然是1,n2的值已经改变,这说明std::bind使用的是参数的拷贝而不是引用,这也就是为什么C++11要引入std::ref和std::cref的原因了,接下来分析std::ref的实现(std::cref不作分析,因为std::cref和std::ref唯一的差别只是引用变成了const而已)。

2.std::reference_wrapper介绍

        std::reference_wrapper是一个模板类,用于包装引用,使其能够在容器中存储或以引用的形式传递。它提供类似引用的语法,并且可以与标准容器一起使用,因为容器无法直接存储引用。                

        其实使用std::ref时,后台真正起作用的关键类是std::reference_wrapper,它才是真正包装引用的类。它的实现如下:

template <class _Ty>
class reference_wrapper
#if !_HAS_CXX20: public _Weak_types<_Ty>
#endif // !_HAS_CXX20
{
public:static_assert(is_object_v<_Ty> || is_function_v<_Ty>,"reference_wrapper<T> requires T to be an object type or a function type.");using type = _Ty;template <class _Uty, enable_if_t<conjunction_v<negation<is_same<_Remove_cvref_t<_Uty>, reference_wrapper>>,_Refwrap_has_ctor_from<_Ty, _Uty>>,int> = 0>_CONSTEXPR20 reference_wrapper(_Uty&& _Val) noexcept(noexcept(_Refwrap_ctor_fun<_Ty>(_STD declval<_Uty>()))) {_Ty& _Ref = static_cast<_Uty&&>(_Val);_Ptr      = _STD addressof(_Ref);}_CONSTEXPR20 operator _Ty&() const noexcept {return *_Ptr;}_NODISCARD _CONSTEXPR20 _Ty& get() const noexcept {return *_Ptr;}private:_Ty* _Ptr{};public:template <class... _Types>_CONSTEXPR20 auto operator()(_Types&&... _Args) constnoexcept(noexcept(_STD invoke(*_Ptr, static_cast<_Types&&>(_Args)...))) // strengthened-> decltype(_STD invoke(*_Ptr, static_cast<_Types&&>(_Args)...)) {return _STD invoke(*_Ptr, static_cast<_Types&&>(_Args)...);}
};

从源代码中可以看出以下几点信息:

1)有一个类成员_Ptr,类型为所引用类型的指针,用于存储实际对象的地址

2)std::reference_wrapper的构造函数中的限制条件:

conjunction_v<negation<is_same<_Remove_cvref_t<_Uty>, reference_wrapper>>,_Refwrap_has_ctor_from<_Ty, _Uty>>

_Remove_cvref_t : 把类型中的引用、const、volatile去掉

is_same: 判断两个数据类型是否一样

negation: 逻辑非

conjunction_v : 逻辑与

_Refwrap_has_ctor_from定义如下:

// CLASS TEMPLATE reference_wrapper
template <class _Ty>
void _Refwrap_ctor_fun(_Identity_t<_Ty&>) noexcept;
template <class _Ty>
void _Refwrap_ctor_fun(_Identity_t<_Ty&&>) = delete;template <class _Ty, class _Uty, class = void>
struct _Refwrap_has_ctor_from : false_type {};template <class _Ty, class _Uty>
struct _Refwrap_has_ctor_from<_Ty, _Uty, void_t<decltype(_Refwrap_ctor_fun<_Ty>(_STD declval<_Uty>()))>> : true_type {};

        从template <class _Ty> void _Refwrap_ctor_fun(_Identity_t<_Ty&&>) = delete;这行代码可以看出_Refwrap_has_ctor_from是拒绝右值的,所以我们可以看出std::reference_wrapper的构造是拒绝右值引用的

        接下来在std::reference_wrapper构造函数里面,形参对应的是被持有对象的左值引用类型,其接受ref函数接受的左值引用变量_Ref。通过addressof(_Ref)函数取出该变量的地址,将这个地址信息存入成员变量_Ptr中。

3)重载类型转换标识符  operator _Ty&()和_Ty& get(),这个操作符提供了该对象类型到被包装类型的左值引用类型的类型转换。从而可以让这个对象像未包装前的类型一样去使用。

4)重载函数调用操作符 operator() ,这对应的被包装对象是可调用对象如lambda的数据类型的版本。基本原理和对值类型变量的原理一致

5)std::reference_wrapper与普通引用最大的不同是:该引用可以拷贝或赋值

        从上面的代码中,可以看出来。为了保证引用类型在经过函数模板或者类模板中的值传递过程中可以保持引用信息。这里面采用将传入变量包装成另外一个新的对象,在这个新的对象中持有被包装对象的地址信息。在函数模板和类模板的值传递过程中,对这个新的对象进行值传递,其内部的被包装的对象地址信息可以得到保存。在函数模板或者内模板内部使用这个新的对象的时候,可以通过重载的类型转换函数将被包装变量的地址信息转换还原成相应的引用,对这个引用进行操作。从而达到操作外部变量的作用。

        下面演示将 std::reference_wrapper 作为引用的容器使用,这令使用多重索引访问同一容器称为可能。

#include <algorithm>
#include <functional>
#include <iostream>
#include <list>
#include <numeric>
#include <random>
#include <vector>void println(auto const rem, std::ranges::range auto const& v)
{for (std::cout << rem; auto const& e : v)std::cout << e << ' ';std::cout << '\n';
}int main()
{std::list<int> l(10);std::iota(l.begin(), l.end(), -4);// 不能在 list 上用 shuffle(要求随机访问),但能在 vector 上使用它std::vector<std::reference_wrapper<int>> v(l.begin(), l.end());std::ranges::shuffle(v, std::mt19937{std::random_device{}()});println("list 的内容: ", l);println("list 的内容,通过经混洗的 vector 所见: ", v);std::cout << "倍增初始化式列表中的值...\n";std::ranges::for_each(l, [](int& i) { i *= 2; });println("list 的内容,通过经混洗的 vector 所见: ", v);
}

输出:

list 的内容: -4 -3 -2 -1 0 1 2 3 4 5
list 的内容,通过经混洗的 vector 所见: -1 2 -2 1 5 0 3 -3 -4 4
倍增初始化式列表中的值...
list 的内容,通过经混洗的 vector 所见: -2 4 -4 2 10 0 6 -6 -8 8

3.std::ref原理分析

// FUNCTION TEMPLATES ref AND cref
template <class _Ty>
_NODISCARD _CONSTEXPR20 reference_wrapper<_Ty> ref(_Ty& _Val) noexcept {return reference_wrapper<_Ty>(_Val);
}template <class _Ty>
void ref(const _Ty&&) = delete;template <class _Ty>
_NODISCARD _CONSTEXPR20 reference_wrapper<_Ty> ref(reference_wrapper<_Ty> _Val) noexcept {return _STD ref(_Val.get());
}

有了之前的std::reference_wrapper介绍,std::ref的理解就简单很多了。从源代码中可以看出以下几点信息:

1)std::ref是一个模板函数,返回值是模板类std::reference_wrapper
2)从第二个函数可以看到,std::ref不允许传递右值引用参数,即无法包装右值引用传递的值
std::ref的传入参数可以是一个普通的引用,也可以是另外一个std::reference_wrapper对象

示例如下:

#include <iostream>
#include <functional>void func(int& value) {value *= 2;
}int main() {int number = 42;auto refNumber = std::ref(number);func(refNumber);  // 使用可修改的引用作为参数std::cout << "func Value: " << number << std::endl;return 0;
}

        调用std::ref返回一个类型为std::reference_wrapper的refNumber, 调用func前隐式转换成这样int&类型的参数, 然后就能顺利的改变 refNumber 的值了。

4.std::cref原理分析

template <class _Ty>
_NODISCARD _CONSTEXPR20 reference_wrapper<const _Ty> cref(const _Ty& _Val) noexcept {return reference_wrapper<const _Ty>(_Val);
}template <class _Ty>
void cref(const _Ty&&) = delete;template <class _Ty>
_NODISCARD _CONSTEXPR20 reference_wrapper<const _Ty> cref(reference_wrapper<_Ty> _Val) noexcept {return _STD cref(_Val.get());
}

        std::cref只是在std::ref基础上加了一个const,表示它返回的这个引用不能修改值。其它都一样。可以在需要引用的地方使用。这在函数参数传递中特别有用,因为它允许我们在不进行拷贝的情况下传递常量对象,同时保持引用的语义。

        示例如下:

#include <iostream>
#include <functional>void printValue(const int& value) {std::cout << "Value: " << value << std::endl;
}int main() {int number = 42;auto crefNumber = std::cref(number);printValue(crefNumber);  // 使用常量引用传递参数return 0;
}

5.总结

        总的来说,std::ref 和 std::cref 是用于创建引用包装器的有用工具,但它们通常需要与其他技术(如 std::bind 或 lambda 表达式)结合使用,让它展现出和普通引用类似的效果,以便与 C++ 的算法库等接口兼容。

std::reference_wrapper - cppreference.com

std::ref, std::cref - cppreference.com

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/11672.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

面试题:调整数字顺序,使奇数位于偶数前面

题目&#xff1a; 输入一个整数数组&#xff0c;实现一个函数&#xff0c;来调整该数组中数字的顺序 使得所有奇数位于数组的前半部分&#xff0c;所有偶数位于数组的后半部分 算法1&#xff1a; 利用快速排序的一次划分思想&#xff0c;从2端往中间遍历 时间复杂度&#x…

C++ | Leetcode C++题解之第88题合并两个有序数组

题目&#xff1a; 题解&#xff1a; class Solution { public:void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {int p1 m - 1, p2 n - 1;int tail m n - 1;int cur;while (p1 > 0 || p2 > 0) {if (p1 -1) {cur nums2[p2-…

Alist + RaiDrive-Nas挂载云盘(Quark)

Alist RaiDrive-Nas挂载云盘(Quark) Alist download Alist document RaiDriver download — https://www.raidrive.com/ nssm download nssm document nssm specification 配置 Alist 下载适合的Alist版本后&#xff0c;启动服务&#xff1b; 需使用命令符。 在完成解…

数字水印 | Python 基于离散小波变换 DWT 的图像水印嵌入(上)

&#x1f34d;原文&#xff1a; 基于 dwt (离散小波变换) 实现彩色图像水印嵌入部分_1.0 &#x1f34d;写在前面&#xff1a; 本文在原文的基础上进行了代码补全。 正文 本文的内容主要为&#xff1a;水印图像经过 A r n o l d \mathsf{Arnold} Arnold 置乱算法后&#xff0…

vue+springboot用户注销功能

vue文件前端 <el-button type"warning" plain click"handleDeletion">注 销</el-button> // 注销 const handleDeletion (userName) > {ElMessageBox.confirm(注销该用户所有信息后无法恢复&#xff0c;您确认注销吗?, 注销确认, { type…

实现日期类

日期类的实现主要是去学习使用operator的 日期类就是计算日期之间的天数&#xff0c;日期与&#xff08;日期&#xff0c;天数&#xff09;的相加减 比如日常生活中我们可以计算日期加天数&#xff0c;日期减天数&#xff0c;日期减日期&#xff0c; 但没有日期加日期的说法 日…

M-有效算法

在赛场上&#xff0c;脑子就两个字“二分”&#xff0c;一点思路都没&#xff0c;完全不知道二分谁&#xff0c;怎么二分&#xff0c;从哪入手。隐隐约约也知道要变换公式&#xff0c;可惜没坚持这个想法。脑子里全是把k分离出来&#xff0c;赛后看了题解才知道&#xff0c;应该…

LeetCode 力扣题目:买卖股票的最佳时机 IV

❤️❤️❤️ 欢迎来到我的博客。希望您能在这里找到既有价值又有趣的内容&#xff0c;和我一起探索、学习和成长。欢迎评论区畅所欲言、享受知识的乐趣&#xff01; 推荐&#xff1a;数据分析螺丝钉的首页 格物致知 终身学习 期待您的关注 导航&#xff1a; LeetCode解锁100…

MQTT学习(二)

订阅主题和订阅确认 SUBSCRIBE——订阅主题 之前的CONNECT报文&#xff0c;分为 固定报头&#xff1a;必须存在&#xff0c;用于描述报文信息。里面有指出什么类型的报文&#xff0c;报文的等级。可变报头&#xff1a;不一定存在。主要看什么样子类型的报文。有效载荷部分&a…

LoRA Land: 310个经微调的大语言模型可媲美GPT-4

摘要 低秩自适应 (LoRA) 已成为大语言模型 (LLM) 参数有效微调 (PEFT) 中最广泛采用的方法之一。LoRA 减少了可训练参数的数量和内存使用,同时达到了与全面微调相当的性能。该研究旨在评估在实际应用中训练和服务使用 LoRA 微调的 LLM 的可行性。首先,该研究测量了在 10 个基础…

js基础-数组-事件对象-日期-本地存储

一、大纲 一、获取元素位置 在JavaScript中&#xff0c;获取一个元素在页面上的位置可以通过多种方法实现。以下是一些常见的方法&#xff1a; getBoundingClientRect() getBoundingClientRect() 方法返回元素的大小及其相对于视口的位置。它提供了元素的left、top、right和bo…

Vue的学习 —— <vue响应式基础>

目录 前言 正文 单文件组件 什么是单文件组件 单文件组件使用方法 数据绑定 什么是数据绑定 数据绑定的使用方法 响应式数据绑定 响应式数据绑定的使用方法 ref() 函数 reactive()函数 toRef()函数 toRefs()函数 案例练习 前言 Vue.js 以其高效的数据绑定和视图…

探索大语言模型代理(Agent):研究背景、通用框架与未来展望

引言 近年来&#xff0c;随着人工智能技术的飞速发展&#xff0c;大语言模型&#xff08;Large Language Models, LLMs&#xff09;在智能代理&#xff08;Agent&#xff09;领域中的应用已成为研究的热点。这些代理不仅能够模拟人类的认知过程&#xff0c;还能在复杂的社会环…

CNN/TCN/LSTM/BiGRU-Attention到底哪个模型效果最好?注意力机制全家桶来啦!

​ 声明&#xff1a;文章是从本人公众号中复制而来&#xff0c;因此&#xff0c;想最新最快了解各类智能优化算法及其改进的朋友&#xff0c;可关注我的公众号&#xff1a;强盛机器学习&#xff0c;不定期会有很多免费代码分享~ 目录 数据介绍 效果展示 原理简介 代…

数字人解决方案——AniTalker声音驱动肖像生成生动多样的头部说话视频算法解析

1.概述 AniTalker是一款先进的AI驱动的动画生成工具&#xff0c;它超越了简单的嘴唇同步技术&#xff0c;能够精准捕捉并再现人物的面部表情、头部动作以及其他非言语的微妙动态。这不仅意味着AniTalker能够生成嘴型精准同步的视频&#xff0c;更重要的是&#xff0c;它还能够…

使用Dockerfile配置Springboot应用服务发布Docker镜像-16

创建Docker镜像 springboot-docker模块 这个应用可以随便找一个即可&#xff0c;这里不做详细描述了。 pom.xml 依赖版本可参考 springbootSeries 模块中pom.xml文件中的版本定义 <dependencies><dependency><groupId>com.alibaba.cloud</groupId>…

[数据集][图像分类]杂草分类数据集17509张9类别

数据集格式&#xff1a;仅仅包含jpg图片&#xff0c;每个类别文件夹下面存放着对应图片 图片数量(jpg文件个数)&#xff1a;17509 分类类别数&#xff1a;9 类别名称:["chineseapple","lantana","negatives","parkinsonia","part…

48-Qt控件详解:Buttons Containers2

一 Group Box:组合框 #include "widget.h"#include<QGroupBox> #include<QRadioButton> #include<QPushButton> #include<QVBoxLayout>//可以在水平方向和垂直方向进行排列的控件&#xff0c;QHBoxLayout/QVBoxLayout #include <QGridLa…

解决宝塔Nginx和phpMyAdmin配置端口冲突问题

问题描述 在对基于宝塔面板的 Nginx 配置文件进行端口修改时&#xff0c;我注意到 phpMyAdmin 的端口配置似乎也随之发生了变化&#xff01; 解决方法 官方建议在处理 Nginx 配置时&#xff0c;应避免直接修改默认的配置文件&#xff0c;以确保系统的稳定性和简化后续的维护…

大数据可视化实验三——数据可视化工具使用

目录 一、实验目的... 1 二、实验环境... 1 三、实验内容... 1 1. 下载并安装Tableau软件.. 1 2. 使用HTML5绘制Canvas图形.. 2 3. 使用HTML5编写SVG 图形... 5 4. 使用R 语言编写可视化实例.. 7 四、总结与心得体会... 7 五、思考问题... 8 一、实验目的 1&#xff…