C++:引用,万能引用,引用折叠,std::forward一次带你搞明白

前言胡扯

 

我上一篇博客讲了move,跟这篇博客有很强的关联性,这篇博客我也会提到很多上篇博客讲的东西,希望大家看到感兴趣想多了解的话,可以去看看我的往期博客,会有所收获的。

一,万能引用

        万能引用只有一种形式:T&&,不能携带任何的修饰符。

        我上篇博客提到了一句很重要的话:函数的形参永远是左值。

        

#include<iostream>
#include<type_traits>
template<typename T>
void fun(T&& t)
{
	std::cout << std::boolalpha;
	std::cout << std::is_reference_v<T> << std::endl;
	std::cout << std::is_const_v<T> << std::endl;
}
int main()
{
	const int a = 5;
	fun(5);
	fun(a);//T 是 const int&
}

 

        我们能得到第二个是引用,第一个不是引用,第二个是左值,第一个不是左值 。

        我们看上面的例子才能知道通过万能引用推导规则,知道什么时候是右值,什么时候是左值:

        左值————>左值引用(加上引用):看到带引用的就知道它原来是左值

        右值————>不带引用(把引用去掉) :看到不带引用的就知道它原来是右值

        我们发现我们传进去的const的修饰符也消失了,但实际上并不是如此,这里就要有顶层const和底层const的区别了,这里我不过多介绍,可以看我往期博客有专门介绍const,引用是只有底层const的,不存在顶层const你只能const int&这样写引用对吧,而is_const_v这个方法它只能检测到顶层const,所以输出了false。以上就是万能引用的推导规则。

二,引用折叠

        引用折叠只能应用于推导的语境下。(如:模板实例化,auto,decltype等)

        右值引用加上右值引用等于右值引用(&&+&&==&&),除了这个外其他的所有形式都为左值引用。

        记住这两句话,你就可以解决引用折叠的百分之一百的问题了我们接着来看这个例子

        

#include<iostream>
#include<type_traits>
template<typename T>
void fun(T&& t)
{
	std::cout << std::boolalpha;
	std::cout << std::is_reference_v<T> << std::endl;
	std::cout << std::is_const_v<T> << std::endl;
}
int main()
{
	const int a = 5;
	fun(5);//
	fun(a);//t=const int& + &&==const int&
}

         所以能看出大概是这么个情况,所以推导为了左值引用。

三,std::forward

        forward不进行任何转发,是在某个特定情况下,进行一个强制转换 (static_cast)

        而我上篇博客讲的move 是强制转换为右值引用,看清楚forward和move的区别。当我们有一个复杂的对象时,我们更希望可以移动它而不是拷贝它浪费性能,所以我们希望它们变成右值,就得判断传入的形参为右值还是左值,通过万能引用的推导规则我能知道,左值为左值引用,右值不带引用,这样就能知道该调用移动语句还是复制语句,forward就是基于上面实现的,我们拿上一篇的博客举例。

#include<iostream>
#include<type_traits>
#include<vector>
class Vector
{
private:
	int x, y, z;
public:
	Vector(int x, int y, int z) :x(x), y(y), z(z) {}
};
std::vector<Vector> vec;
template<typename T>
void fun(T&& t)
{
	vec.push_back(std::forward<T>(t));
}
int main()
{
	const Vector v0(1, 2, 3);
	fun(v0);
	fun(Vector(2, 3, 4));
}

他们分别会调用

    _CONSTEXPR20_CONTAINER void push_back(const _Ty& _Val) { // insert element at end, provide strong guarantee
        emplace_back(_Val);
    }

    _CONSTEXPR20_CONTAINER void push_back(_Ty&& _Val) {
        // insert by moving into element at end, provide strong guarantee
        emplace_back(_STD move(_Val));
    }

左右值会分别调用上面的复制语句和移动语义,我们最后看一下forward的源码吧

template <class _Ty>
_NODISCARD constexpr _Ty&& forward(
    remove_reference_t<_Ty>& _Arg) noexcept { // forward an lvalue as either an lvalue or an rvalue
    return static_cast<_Ty&&>(_Arg);
}

template <class _Ty>
_NODISCARD constexpr _Ty&& forward(remove_reference_t<_Ty>&& _Arg) noexcept { // forward an rvalue as an rvalue
    static_assert(!is_lvalue_reference_v<_Ty>, "bad forward call");
    return static_cast<_Ty&&>(_Arg);
}

我们看到了他是个重载函数,通过万能引用和引用折叠,返还出我们的值,我们传入的值无非就是,带引用和不带引用两种,第二个它加了个断言用于不加引用的版本以防出错。我们看到了std::forward很简单,当传入参数为不是引用或为右值引用我就强转为右值引用,其他强转为左值引用。

结语

万能引用,引用折叠和完美转发到此也就讲完了,如果觉得会有所收获的话,可以看看博主的往期博客一定会有所收获顺便关注下博主,当然以上内容哪里出错了的话,还是麻烦大家多来指正,好的谢谢大家看到这里!!!!

  • 5
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
C++中,引用折叠指的是在使用引用类型时,根据一定的规则将不同类型的引用组合成为新的引用类型的过程。引用折叠的规则如下: 1. 对于右值引用(Rvalue Reference,T&&),无论与左值引用还是右值引用相结合,都会得到右值引用类型。 2. 对于左值引用(Lvalue Reference,T&),在不使用模板特化的情况下,左值引用与左值引用相结合得到左值引用,右值引用与左值引用结合得到右值引用类型。 3. 在使用模板特化的情况下,引用折叠规则会发生变化。 下面通过代码来说明引用折叠的用法和效果: ```cpp void foo(int& x) { std::cout << "foo(int&)" << std::endl; } void foo(int&& x) { std::cout << "foo(int&&)" << std::endl; } template<typename T> void bar(T&& x) { foo(std::forward<T>(x)); } int main() { int a = 42; bar(a); // 输出 foo(int&) bar(42); // 输出 foo(int&&) return 0; } ``` 在上面的代码中,函数`foo`有两个重载,分别接受一个左值引用和一个右值引用。函数`bar`接受一个通用引用(Universal Reference),并将其转发给`foo`进行调用。在调用`bar`时,如果传入一个左值,那么`T`会被推导为左值引用类型,此时`std::forward<T>(x)`返回的是一个左值引用。如果传入一个右值,那么`T`会被推导为右值引用类型,此时`std::forward<T>(x)`返回的是一个右值引用。这样,我们就可以在`bar`函数中对传入的参数进行完美转发,同时保留其原本的左值或右值属性。 总之,引用折叠C++11引入的一种重要特性,它可以简化模板编程中的引用类型推导,同时也为许多新特性的实现提供了基础。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

KNGG

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值