首页 > 编程语言 >【重学C++】05 | 说透右值引用、移动语义、完美转发(下)

【重学C++】05 | 说透右值引用、移动语义、完美转发(下)

时间:2023-05-29 11:47:47浏览次数:60  
标签:std 透右值 05 factory C++ arg shared ptr

文章首发

【重学C++】05 | 说透右值引用、移动语义、完美转发(下)

引言

大家好,我是只讲技术干货的会玩code,今天是【重学C++】的第五讲,在第四讲《【重学C++】04 | 说透右值引用、移动语义、完美转发(上)》中,我们解释了右值和右值引用的相关概念,并介绍了C++的移动语义以及如何通过右值引用实现移动语义。今天,我们聊聊右值引用的另一大作用 -- 完美转发

什么是完美转发

假设我们要写一个工厂函数,该工厂函数负责创建一个对象,并返回该对象的智能指针。

template<typename T, typename Arg> 
std::shared_ptr<T> factory_v1(Arg arg)
{
	return std::shared_ptr<T>(new T(arg));
}

class X1 {
public:
	int* i_p;
	X(int a) {
		i_p = new int(a);
	}
}

对于类X的调用方来说,auto x1_ptr = factory_v1<X1>(5); 应该与auto x1_ptr = std::shared_ptr<X>(new X1(5))是完全一样的。

也就是说,工厂函数factory_v1对调用者是透明的。要达到这个目的有两个前提:

  1. 传给factory_v1的入参arg能够完完整整(包括引用属性、const属性等)得传给T的构造函数。
  2. 工厂函数factory_v1没有额外的副作用。

这个就是C++的完美转发。

单看factory_v1应用到X1貌似很"完美",但既然是工厂函数,就不能只满足于一种类对象的应用。假设我们有类X2。定义如下

class X2 {
public:
	X2(){}
	X2(X2& rhs) {
		std::cout << "copy constructor call" << std::endl;
	}
}

现在大家再思考下面代码:

X2 x2 = X2();

auto x2_ptr1 = factory_v1<X2>(x2);
// output:
// copy constructor call
// copy constructor call

auto x2_ptr2 = std::shared_ptr<X2>(x2)
// output:
// copy constructor call

可以发现,auto x2_ptr1 = factory_v1<X2>(x2);auto x2_ptr2 = std::shared_ptr<X2>(x2)多了一次拷贝构造函数的调用。

为什么呢?很简单,因为factory_v1的入参是值传递,所以x2在传入factory_v1时,会调用一次拷贝构造函数,创建arg。很直接的办法,把factory_v1的入参改成引用传递就好了,得到factory_v2

template<typename T, typename Arg> 
std::shared_ptr<T> factory_v2(Arg& arg)
{
	return std::shared_ptr<T>(new T(arg));
}

改成引用传递后,auto x1_ptr = factory_v2<X1>(5);又会报错了。因为factory_v2需要传入一个左值,但字面量5是一个右值。

方法总比困难多,我们知道,C++的const X& 类型参数,既能接收左值,又能接收右值,所以,稍加改造,得到factory_v3

template<typename T, typename Arg> 
std::shared_ptr<T> factory_v3(const Arg& arg)
{
	return std::shared_ptr<T>(new T(arg));
}

factory_v3还是不够"完美", 再看看另外一个类X3

class X3 {
public:
	X3(){}
	X3(X3& rhs) {
		std::cout << "copy constructor call" << std::endl;
	}

	X3(X3&& rhs) {
		std::cout << "move constructor call" << std::endl;
	}
}

再看看以下使用例子

auto x3_ptr1 = factory_v3<X3>(X3());
// output
// copy constructor call

auto x3_ptr2 = std::shared_ptr<X3>(new X3(X3()));
// output
// move constructor call

通过上一节我们知道,有名字的都是左值,所以factory_v3永远无法调用到T的移动构造函数。所以,factory_v3还是不满足完美转发。

特殊的类型推导 - 万能引用

给出完美转发的解决方案前,我们先来了解下C++中一种比较特殊的模版类型推导规则 - 万能引用。

// 模版函数签名
template <typename T>
void foo(ParamType param);

// 应用
foo(expr);

模版类型推导是指根据调用时传入的expr,推导出模版函数fooParamTypeparam的类型。

类型推导的规则有很多,大家感兴趣可以去看看《Effective C++》[1],这里,我们只介绍一种比较特殊的万能引用。 万能引用的模版函数格式如下:

template<typename T>
void foo(T&& param);

万能引用的ParamTypeT&&,既不能是const T&&,也不能是std::vector<T>&&

万能引用的规则有三条:

  1. 如果expr是左值,Tparam都会被推导成左值引用。
  2. 如果expr是右值,T会被推导成对应的原始类型,param会被推导成右值引用(注意,虽然被推导成右值引用,但由于param有名字,所以本身还是个左值)。
  3. 在推导过程中,expr的const属性会被保留下来。

看下面示例

template<typename T>
void foo(T&& param);
// x是一个左值
int x=27;
// cx是带有const的左值
const int cx = x;
// rx是一个左值引用
const int& rx = cx;

// x是左值,所以T是int&,param类型也是int&
foo(x);

// cx是左值,所以T是const int&,param类型也是const int&
foo(cx);

// rx是左值,所以T是const int&,param类型也是const int&
foo(rx);

// 27是右值,所以T是int,param类型就是int&&
foo(27);

std::forward实现完美转发

到此,完美转发的前置知识就已经讲完了,我们看看C++是如何利用std::forward实现完美转发的。

template<typename T, typename Arg> 
std::shared_ptr<T> factory_v4(Arg&& arg)
{ 
  return std::shared_ptr<T>(new T(std::forward<Arg>(arg)));
}

std::forward的定义如下

template<class S>
S&& forward(typename remove_reference<S>::type& a) noexcept
{
  return static_cast<S&&>(a);
}

传入左值

X x;
auto a = factory_v4<A>(x);

根据万能引用的推导规则,factory_v4中的Arg会被推导成X&。这个时候factory_v4std::forwrd等价于:

shared_ptr<A> factory_v4(X& arg)
{ 
  return shared_ptr<A>(new A(std::forward<X&>(arg)));
}

X& std::forward(X& a) 
{
  return static_cast<X&>(a);
}

这个时候传给A的参数类型是X&,即调用的是拷贝构造函数A(X&)。符合预期。

传入右值

X createX();
auto a = factory_v4<A>(createX());

根据万能引用推导规则,factory_v4中的Arg会被推导成X。这个时候factory_v4std::forwrd等价于:

shared_ptr<A> factory_v4(X&& arg)
{ 
  return shared_ptr<A>(new A(std::forward<X>(arg)));
}

X&& forward(X& a) noexcept
{
  return static_cast<X&&>(a);
}

此时,std::forward作用与std::move一样,隐藏掉了arg的名字,返回对应的右值引用。这个时候传给A的参数类型是X&&,即调用的是移动构造函数A(X&&),符合预期。

总结

这篇文章,我们主要是继续第四讲的内容,一步步学习了完美转发的概念以及如何使用右值解决参数透传的问题,实现完美转发。

[1] https://github.com/CnTransGroup/EffectiveModernCppChinese/blob/master/src/1.DeducingTypes/item1.md

END

【往期推荐】

【重学C++】01| C++ 如何进行内存资源管理?

【重学C++】02 | 脱离指针陷阱:深入浅出 C++ 智能指针

【重学C++】03 | 手撸C++智能指针实战教程

【重学C++】04 | 说透C++右值引用、移动语义、完美转发(上)

标签:std,透右值,05,factory,C++,arg,shared,ptr
From: https://www.cnblogs.com/huiwancode/p/17440005.html

相关文章

  • 【2023-05-27】要当“人中人”
    20:00人只要还有一些输出的力量,生命至少不是徒劳的消耗吧。如果我什么都不做的话,会觉得这一生好像就是一场浩劫。                                                 ——......
  • 【2023-05-26】当务之重
    20:00我不相信没有种子植物也能发芽,我心中有对种子的信仰。让我相信你有一颗种子,我等待奇迹。                                                 ——梭罗在我的成长的......
  • MxDraw(在线CAD,H5开发CAD) 2023.05.23更新
    1. 完善了CAD在线编辑功能2. 增加addControlsEvent函数3. 修改Linux下,转换大图纸时,有部分照出当前视范围 ,就不显示问题4. 在CAD打开图纸增加多线程支持5. 修改片元程序编译加载失败问题6. 修改iconv 在centos7上不对问题7. 增加对对象的引用计数判断,防止对象被释放......
  • C++“高级程序设计实践(C++)”课程设计任务书[2023-05-28]
    C++“高级程序设计实践(C++)”课程设计任务书[2023-05-28]“高级程序设计实践(C++)”课程设计任务书一.课程设计的目的与任务本设计是课程——《面向对象编程(C++)》的一个关键实践环节。它是根据教学计划的要求,在教师的指导下,对学生实施程序设计训练的必要过程,是对前期课堂学习内......
  • C++模板元编程:用代码生成代码的黑科技
    介绍模板元编程在C++中使用模板元编程示例一:元函数示例二:元数据计算示例三:元编程递归总结让我们开始吧。介绍模板元编程在C++编程中,元编程是一种特殊的技术,所谓的元编程就是在编写代码时,使用代码来生成代码。与传统的程序设计思路不同,元编程充分利用了C++模板的特性。采用元编程技......
  • 总结20230528
    代码时间(包括上课)2h代码量(行):50行博客数量(篇):1篇相关事项:1、今天直接凌晨五点才到宿舍,连夜整的无人机,为后天的比赛准备。2、今天上午上的计算机网络,由于比赛冲突,请假了没去上。3、今天下午的web上机也没去,正好赶上web报告也写完。4、晚上也是连夜通宵整的无人机。......
  • 2023-05-28 TypeScript学习记录(长更)
    概述:TypeScript(下称ts),js的超集,在js基础上进行了扩展并且新增了一些类型;不能被浏览器直接识别,需要编译为js才能被执行。为什么使用ts,而不是js:js语法的定义相对不够严谨,变量没有约束,而ts在js一些不足的地方进行了优化,使写法变得严谨也更为复杂起来。ts安装:npminstall-gtypescri......
  • 2023-05-28:为什么Redis单线程模型效率也能那么高?
    2023-05-28:为什么Redis单线程模型效率也能那么高?答案2023-05-28:1.C语言实现,效率高C语言程序运行速度快,因为其相较于其他高级语言更加接近底层机器。由于C语言直接操作内存,不会像其他语言那样依赖虚拟机或垃圾回收机制等中间层,从而能够实现更高的执行效率。2.单线程的优势Redi......
  • 2023-05-25 EMC
    andcanintheinjectionclampat750mm.AndnowI'veperformedathresholdanalysisonexactlythisfailposition.Andthisisnowinthepresentationonthenextslide.Andyoucanseewewentup1dBmicroampAnd102dBmicroampsiscategorytwo......
  • 2023-05-28:为什么Redis-单线程模型效率也能那么高?
    2023-05-28:为什么Redis-单线程模型效率也能那么高?答案2023-05-28:1.C语言实现,效率高C语言程序运行速度快,因为其相较于其他高级语言更加接近底层机器。由于C语言直接操作内存,不会像其他语言那样依赖虚拟机或垃圾回收机制等中间层,从而能够实现更高的执行效率。2.单线程的优势Redis采用......