move std 函数 示例_c++ 之 std::move 原理实现与用法总结

在C++11中,标准库在中提供了一个有用的函数std::move,std::move并不能移动任何东西,它唯一的功能是将一个左值强制转化为右值引用,继而可以通过右值引用使用该值,以用于移动语义。从实现上讲,std::move基本等同于一个类型转换:static_cast(lvalue);

C++ 标准库使用比如vector::push_back 等这类函数时,会对参数的对象进行复制,连数据也会复制.这就会造成对象内存的额外创建, 本来原意是想把参数push_back进去就行了,通过std::move,可以避免不必要的拷贝操作。

std::move是将对象的状态或者所有权从一个对象转移到另一个对象,只是转移,没有内存的搬迁或者内存拷贝所以可以提高利用效率,改善性能.。

对指针类型的标准库对象并不需要这么做.

用法:

原lvalue值被moved from之后值被转移,所以为空字符串.

1 //摘自https://zh.cppreference.com/w/cpp/utility/move

2 #include

3 #include

4 #include

5 #include

6 intmain()7 {8 std::string str = "Hello";9 std::vector<:string>v;10 //调用常规的拷贝构造函数,新建字符数组,拷贝数据

11 v.push_back(str);12 std::cout << "After copy, str is \"" << str << "\"\n";13 //调用移动构造函数,掏空str,掏空后,最好不要使用str

14 v.push_back(std::move(str));15 std::cout << "After move, str is \"" << str << "\"\n";16 std::cout << "The contents of the vector are \"" << v[0]17 << "\", \"" << v[1] << "\"\n";18 }

输出:

1 After copy, str is "Hello"

2 After move, str is ""

3 The contents of the vector are "Hello", "Hello"

std::move 的函数原型定义

1 template

2 typename remove_reference::type&& move(T&&t)3 {4 return static_cast::type&&>(t);5 }

原型定义中的原理实现:

首先,函数参数T&&是一个指向模板类型参数的右值引用,通过引用折叠,此参数可以与任何类型的实参匹配(可以传递左值或右值,这是std::move主要使用的两种场景)。关于引用折叠如下:

公式一)X& &、X&& &、X& &&都折叠成X&,用于处理左值

1 string s("hello");2 std::move(s) => std::move(string& &&) => 折叠后 std::move(string&)3 此时:T的类型为string&

4 typename remove_reference::type为string5 整个std::move被实例化如下6 string&& move(string& t) //t为左值,移动后不能在使用t

7 {8     //通过static_cast将string&强制转换为string&&

9     return static_cast(t);10 }

公式二)X&& &&折叠成X&&,用于处理右值

1 std::move(string("hello")) => std::move(string&&)2 //此时:T的类型为string3 //remove_reference::type为string4 //整个std::move被实例如下

5 string&& move(string&& t) //t为右值

6 {7     return static_cast(t);  //返回一个右值引用

8 }

简单来说,右值经过T&&传递类型保持不变还是右值,而左值经过T&&变为普通的左值引用.

②对于static_cast<>的使用注意:任何具有明确定义的类型转换,只要不包含底层const,都可以使用static_cast

1 double d = 1;2 void* p = &d;3 double *dp = static_cast p; //正确

4 5 const char *cp = "hello";6 char *q = static_cast(cp); //错误:static不能去掉const性质

7 static_cast(cp); //正确

③对于remove_reference是通过类模板的部分特例化进行实现的,其实现代码如下

1 //原始的,最通用的版本

2 template structremove_reference{3     typedef T type;  //定义T的类型别名为type

4 };5 6 //部分版本特例化,将用于左值引用和右值引用

7 template struct remove_reference //左值引用

8 { typedef T type; }9 10 template struct remove_reference //右值引用

11 { typedef T type; }12 13 //举例如下,下列定义的a、b、c三个变量都是int类型

14 inti;15 remove_refrence::type a;             //使用原版本,

16 remove_refrence::type  b;             //左值引用特例版本

17 remove_refrence::type  b;  //右值引用特例版本

总结:

std::move实现,首先,通过右值引用传递模板实现,利用引用折叠原理将右值经过T&&传递类型保持不变还是右值,而左值经过T&&变为普通的左值引用,以保证模板可以传递任意实参,且保持类型不变。然后我们通过static_cast<>进行强制类型转换返回T&&右值引用,而static_cast之所以能使用类型转换,是通过remove_refrence::type模板移除T&&,T&的引用,获取具体类型T。

参考链接:

https://blog.csdn.net/fengbingchun/article/details/52558914

https://blog.csdn.net/cpriluke/article/details/79462388

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值