前面两节,说明了右值引用和它的作用。下面通过一个string类的编写,来说明右值引用的使用。

相对于C++98,主要是多了移动构造函数和移动赋值运算符

先给出一个简要的声明:

class String
{
public:
String();
String(const char *s); //转化语义
String(const String &s);
String(String &&s);
~String(); String &operator=(const String &s);
String &operator=(String &&s); friend ostream &operator<<(ostream &os, const String &s)
{
return os << s.data_;
}
private:
char *data_;
};

下面依次实现每个函数。

第一个是默认构造函数:

String::String()
:data_(new char[1])
{
*data_ = 0;
cout << "default" << endl;
}

 

然后是char*版本的构造函数:

String::String(const char *s)
:data_(new char[strlen(s) + 1])
{
::strcpy(data_, s);
cout << "char *" << endl;
}

重点来了,我们提供移动构造函数:

String::String(String &&s)
:data_(s.data_)
{
cout << "move construct" << endl;
s.data_ = NULL; //防止释放data
}

这里最重要的一点就是要把s的data置为NULL,因为s是个右值,马上就要析构。这样就成功实现了偷取s的内容

析构函数:

String::~String()
{
delete[] data_;
}

下面我们提供赋值运算符,这里注意一点:

一是处理自我赋值,二是要返回自身引用。

String &String::operator=(const String &s)
{
if(this != &s)
{
delete[] data_;
data_ = new char[strlen(s.data_) + 1];
::strcpy(data_, s.data_);
}
return *this;
} String &String::operator=(String &&s)
{
if(this != &s)
{
cout << "move assignment" << endl;
delete[] data_;
data_ = s.data_;
s.data_ = NULL;
}
return *this;
}

后面的移动构造函数,依然要把s的data置为NULL。

上面两个函数看似正确,但是没有处理发生异常的情况,如果new时发生异常,但是此时原本的data已经被delete,造成错误。

如何解决?

我们提供一个swap函数:

void String::swap(String &s)
{
std::swap(data_, s.data_);
}

一种好的处理方案是:

String &String::operator=(const String &s)
{
String temp(s);
swap(temp); return *this;
} String &String::operator=(String &&s)
{
String temp(s);
swap(temp); return *this;
}

这样,即使生成temp时发生异常,也对自身没有影响。

注意这里没有处理自我赋值,因为自我赋值发生的情况实际比较少,而之前的代码第一行是delete,则必须处理自我赋值。

上面两个赋值运算符可以直接合为一个:

String &String::operator=(String s)
{
swap(s); return *this;
}

事实上,我们在前面也提到过,除了构造函数之外,X &x和X &&类型的函数,可以合二为一为X x,采用传值。

这样,我们的最后一个实现,保证了异常安全。

 

测试代码:

int main(int argc, char const *argv[])
{
String s("foo");
String s2(s);
//String s3(std::move(String("bar")));
String s3(String("bar")); //编译器优化 直接使用char*
cout << s3 << endl; s3 = s;
s3 = String("hello");
cout << s3 << endl;
s3 = std::move(s2);
cout << s3 << endl; return 0;
}

注意:

String s3(String("bar"));

会被编译器优化为

String s3(“bar”)

可以显式使用:

String s3(std::move(String("bar")));

 

完毕。

C++11之右值引用(三):使用C++11编写string类以及“异常安全”的=运算符的更多相关文章

  1. C++11之右值引用(二):右值引用与移动语义

    上节我们提出了右值引用,可以用来区分右值,那么这有什么用处?   问题来源   我们先看一个C++中被人诟病已久的问题: 我把某文件的内容读取到vector中,用函数如何封装? 大部分人的做法是: v ...

  2. C++ 11的右值引用

    目录 一.问题导入 二.右值和右值引用 2.1 左值(lvalue)和右值(rvalue) 2.2 左值引用和右值引用 总结 参考资料 C++11 引入了 std::move 语义.右值引用.移动构造 ...

  3. C++11 的右值引用

    作者:Tinro链接:https://www.zhihu.com/question/22111546/answer/30801982来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请 ...

  4. C++11之右值引用(一):从左值右值到右值引用

    C++98中规定了左值和右值的概念,但是一般程序员不需要理解的过于深入,因为对于C++98,左值和右值的划分一般用处不大,但是到了C++11,它的重要性开始显现出来. C++98标准明确规定: 左值是 ...

  5. 【C/C++开发】C++11:右值引用和转发型引用

    右值引用 为了解决移动语义及完美转发问题,C++11标准引入了右值引用(rvalue reference)这一重要的新概念.右值引用采用T&&这一语法形式,比传统的引用T&(如 ...

  6. c++11的右值引用、移动语义

    对于c++11来说移动语义是一个重要的概念,一直以来我对这个概念都似懂非懂.最近翻翻资料感觉突然开窍,因此记下.其实搞懂之后就会发现这个概念很简单,并无什么高深的地方. 先说说右值引用.右值一般指的是 ...

  7. c++11之右值引用

    本文大部分来自这里,并不是完全着行翻译,如有不明白的地方请参考原文. 在c++中,创建临时对象的开销对程序的影响一直很大,比如以下这个例子: String getName(){ return “Kia ...

  8. [转载]如何在C++03中模拟C++11的右值引用std::move特性

    本文摘自: http://adamcavendish.is-programmer.com/posts/38190.htm 引言 众所周知,C++11 的新特性中有一个非常重要的特性,那就是 rvalu ...

  9. C++11特性-右值引用

    什么是左值,什么是右值 常见的误区有 = 左边的是左值,右边的是右值. 左值:具有存储性质的对象,即lvalue对象,是指要实际占用内存空间.有内存地址的那些实体对象,例如:变量(variables) ...

随机推荐

  1. Linux中权限(r、w、x)对于目录与文件的意义

    Linux中权限(r.w.x)对于目录与文件的意义 一.权限对于目录的意义 1.首先要明白的是目录主要的内容是记录文件名列表和子目录列表,而不是实际存放数据的地方. 2.r权限:拥有此权限表示可以读取 ...

  2. classes could not be found: - android.support.v7.internal.app.WindowDecorActionBar问题的解决方法

    转载至----http://my.oschina.net/u/2425146/blog/546649?fromerr=aDYrFDVx.仅作个人收藏使用,有转载的朋友请连续原作者,谢谢 刚刚进入stu ...

  3. 求LCA最近公共祖先的离线Tarjan算法_C++

    这个Tarjan算法是求LCA的算法,不是那个强连通图的 它是 离线 算法,时间复杂度是 O(m+n),m 是询问数,n 是节点数 它的优点是比在线算法好写很多 不过有些题目是强制在线的,此类离线算法 ...

  4. 通过OpenGL ES在iOS平台实践增强现实(一)

    http://ios.9tech.cn/news/2013/1108/38495.html 1.本文采用OpenGL ES 1固定渲染管线实现,目标为在设备拍摄到的现实世界中,绘制世界坐标轴,并根据设 ...

  5. 解决provisional headers are shown问题

    前端通过ajax的get请求向后端发送数据,后端死活收不到. ajax代码如下: $.ajax({ url:"/news_list", type:"GET", ...

  6. Java进阶之路,技术要点

    宏观方面 一.JAVA.要想成为JAVA(高级)工程师肯定要学习JAVA.一般的程序员或许只需知道一些JAVA的语法结构就可以应付了.但要成为JAVA(高级)工程师,您要对JAVA做比较深入的研究.您 ...

  7. js面试总结

    <div id="app"> <button onClick="app()">点击1</button> <button ...

  8. HDU 1171 Big Event in HDU【01背包/求两堆数分别求和以后的差最小】

    Big Event in HDU Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) T ...

  9. MyBatis笔记:invalid bound statement (not found)

    maven项目在本地运行的时候没有问题,一旦把war包部署到测试机上就不能运行.查看了一下tomcat日志发现抛出这样的错误:invalid bound statement (not found),后 ...

  10. Cookie和Session在Node.JS中的实践(三)

    Cookie和Session在Node.JS中的实践(三) 前面作者写的COOKIE篇.SESSION篇,算是已经比较详细的说明了两者间的区别.机制.联系了.阅读时间可能稍长,因为作者本身作图也做了不 ...