c++ 之 std::move 原理实现与用法总结 |
您所在的位置:网站首页 › moving和moved的用法 › c++ 之 std::move 原理实现与用法总结 |
转载自:https://blog.csdn.net/p942005405/article/details/84644069/ 在C++11中,标准库在中提供了一个有用的函数std::move,std::move并不能移动任何东西,它唯一的功能是将一个左值强制转化为右值引用,继而可以通过右值引用使用该值,以用于移动语义。从实现上讲,std::move基本等同于一个类型转换:static_cast(lvalue); std::move函数可以以非常简单的方式将左值引用转换为右值引用。(左值 右值 引用 左值引用)概念 https://blog.csdn.net/p942005405/article/details/84644101 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 int main() 7 { 8 std::string str = "Hello"; 9 std::vector v; 10 //调用常规的拷贝构造函数,新建字符数组,拷贝数据 11 v.push_back(str); 12 std::cout std::move(string& &&) => 折叠后 std::move(string& ) 3 此时:T的类型为string& 4 typename remove_reference::type为string 5 整个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的类型为string 3 // remove_reference::type为string 4 //整个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 struct remove_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 int i; 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 https://blog.csdn.net/swartz_lubel/article/details/59620868 |
今日新闻 |
推荐新闻 |
CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3 |