C++ Primer : 第十三章 : 拷贝控制之对象移动
右值引用
int i = 42;
int &r = i; // 正确,r引用i
int&& rr = i; // 错误,不能将右值引用绑定到左值
int &r2 = i * 42; // 错误,i*42是一个右值
const int& r3 = i * 42;
int&& rr2 = i * 42;
int&& rr1 = 42;
int&& rr2 = rr1; // 错误,表达式rr1是左值
int&& rr3 = std::move(rr1);
move旨在告诉编译器,我们希望把一个左值像右值一样使用它。使用move函数就意味着:除了对rr1赋值或销毁之外,我们不能再使用它。使用move之后,我们就不能对源对象的值做任何假设。
移动构造函数和移动赋值运算符
StrVec::StrVec(StrVec&& s) noexcepd : first(s.first), last_end(s.last_end), cap(s.cap){
s.first = s.last_end = s.cap = nullptr; // 将资源的源对象进入安全状态,对其运行析构函数是安全的。
}
移动构造函数类似拷贝构造函数,第一个形参必须是该类类型的一个引用,而其他额外的参数必须有默认实参。
class StrVec {
public:
StrVec(StrVec&&) noexcept;
/*
*/
};
StrVec::StrVec(StrVec&& s) noexcepd : first(s.first), last_end(s.last_end), cap(s.cap){
s.first = s.last_end = s.cap = nullptr; // 将资源的源对象进入安全状态,对其运行析构函数是安全的。
}
StrVec& StrVec::operator = (StrVec&& rhs) noexcept {
if (this != &rhs) {
free();
first = rhs.first;
last_end = rhs.last_end ;
cap = rhs.cap;
rhs.first = rhs.last_end = rhs.cap = nullptr;
}
return *this;
}
- 与拷贝构造函数不同,移动构造函数被定义为删除的函数的条件是:有类成员定义了自己的拷贝构造函数且未定义移动构造函数,或者是有类成员未定义自己的拷贝构造函数但是编译器不能为其合成移动构造函数。移动赋值运算符的情况类似。
- 如果有类成员的移动构造函数或移动赋值运算符被定义为删除的或不可访问的(例如private),则类的移动构造函数或移动赋值运算符被定义为删除的。
- 类似拷贝构造函数,如果类的析构函数被定义为删除的或不可访问的, 则类的移动构造函数被定义为删除的。
- 类似拷贝赋值运算符,如果有类成员是const或是引用,则类的移动赋值运算符被定义为删除的。
右值引用和成员函数
void push_back(const X&);
void push_back(X&&);
我们就可以为动态内存管理类定义另一个版本的push_back:
class StrVec {
public:
void push_back(const std::string&); // 拷贝元素
void push_back(std::string&&); // 移动元素
// 其他定义
};
void StrVec::push_back(const std::string& s) {
chk_n_alloc();
alloc.construct(last_end++, s);
}
void StrVec::push_back(std::string&& s) {
chk_n_alloc();
alloc.construct(last_end++, std::move(s));
}
右值和左值引用函数:
string s1 = "a value", s2 = "another";
auto n = (s1 + s2).find('a');
在此例中,我们在string的右值上调用成员函数,而还有更令我们惊奇的操作:
s1 + s2 = "wow";
我们对一个右值进行了赋值。
class Foo {
public:
Foo& operator = (const Foo&) &; // 只能向可修改的左值赋值
};
Foo& Foo::operator = (const Foo& rhs)& {
// 赋值操作
return *this;
}
引用限定符可以&或&&, 分别指出this可以指向一个左值或右值,类似const限定符,引用限定符只能用于非static函数,而且必须同时出现在声明和定义中。
对于限定 &的函数,我们只能将它用于左值,限定&&的函数,只能用于右值。
class Foo {
public:
Foo someMem() & const;// 错误,const必须在引用限定符之前
Foo anotherMem() const &; // 正确
};
class Foo {
public:
Foo someMem() &;
Foo anotherMem() const; // 错误,必须加上引用限定符
};
C++ Primer : 第十三章 : 拷贝控制之对象移动的更多相关文章
- C++ Primer : 第十三章 : 拷贝控制之拷贝控制和资源管理
定义行为像值的类 行为像值的类,例如标准库容器和std::string这样的类一样,类似这样的类我们可以简单的实现一个这样的类HasPtr. 在实现之前,我们需要: 定义一个拷贝构造函数,完成stri ...
- C++ Primer : 第十三章 : 拷贝控制之拷贝、赋值与销毁
拷贝构造函数 一个构造函数的第一个参数是自身类类型的引用,额外的参数(如果有)都有默认值,那么这个构造函数是拷贝构造函数.拷贝构造函数的第一个参数必须是一个引用类型. 合成的拷贝构造函数 在我们没 ...
- C++ Primer : 第十三章 : 拷贝控制示例
/* Message.h */ #ifndef _MESSAGE_H_ #define _MESSAGE_H_ #include <iostream> #include <strin ...
- [C++ Primer] : 第13章: 拷贝控制
拷贝, 赋值与销毁 当定义一个类时, 我们显示地或隐式地指定在此类型的对象拷贝, 移动, 赋值和销毁时做什么. 一个类通过定义5种特殊的成员函数来控制这些操作, 包括: 拷贝构造函数, 拷贝赋值运算符 ...
- 【C++ Primer 第十三章】4. 拷贝控制示例
拷贝控制示例 #include<iostream> #include<string> #include<set> #include<vector> us ...
- 《C++ Primer》笔记 第13章 拷贝控制
拷贝和移动构造函数定义了当用同类型的另一个对象初始化本对象时做什么.拷贝和移动赋值运算符定义了将一个对象赋予同类型的另一个对象时做什么.析构函数定义了当此类型对象销毁时做什么.我们称这些操作为拷贝控制 ...
- C++Primer 第十三章
//1.当定义一个类时,我们显示地或隐式地指出在此类型的对象(注意这里是此类型的对象,而不包括此类型的指针)拷贝,移动,赋值,销毁时做什么.一个类通过定义五种特殊的成员函数来控制这些操作:拷贝构造函数 ...
- 【c++ Prime 学习笔记】第13章 拷贝控制
定义一个类时,可显式或隐式的指定在此类型对象上拷贝.移动.赋值.销毁时做什么.通过5种成员函数实现拷贝控制操作: 拷贝构造函数:用同类型的另一个对象初始化本对象时做什么(拷贝初始化) 拷贝赋值算符:将 ...
- C++ Primer 5th 第13章 拷贝控制
当一个对象的引用或者指针离开作用域时,析构函数不会执行. 构造函数有初始化部分(初始化列表)和函数体. 析构函数有析构部分和函数,但析构函数的析构部分是隐式的.
随机推荐
- 后勤数据源增量队列Delta Queue(RSA7)中的增量更新区Delta Update、增量重复区Delta Repetition
声明:原创作品,转载时请注明文章来自SAP师太技术博客:( 博/客/园www.cnblogs.com)www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
- js处理iframe中子页面与父页面里面对象的取得的解决方案
[1]子页面取得父页面的dom对象 parent.window.$('#id').val(""); [2]父页面取得子页面的对象 $(window.frames[&qu ...
- python成长之路【第六篇】:python模块--time和datetime
1.时间表现形式 时间戳 (1970年1月1日之后的秒,即:time.time())格式化的时间字符串 (2014-11-11 11:11, 即:time.strftime('%Y-%m- ...
- Logistic 分类器与 softmax分类器
首先说明啊:logistic分类器是以Bernoulli(伯努利) 分布为模型建模的,它可以用来分两种类别:而softmax分类器以多项式分布(Multinomial Distribution)为模型 ...
- [Excel] Excel固定任意行或者任意列
固定第一行第一列:点击B2单元格[以B2为中介点,找你冻结部分的中介点!行列的交叉点!] 例如只想固定第一行,那么请选择A2的单元格 为中介点,A3.A4…… 例如只想固定第一列,那么请选择B1的单元 ...
- bzoj题解汇总(1052~1061)
bzoj1052: 贪心 bzoj1053: DFS bzoj1054: 加深迭代搜索 bzoj1055:区间判定性dp bzoj1056: Treap bzoj1057: 二分,单调队列 / ST表 ...
- 【html5】常见标签使用说明(持续更新)
说明: 所谓常见,是指我在优秀网页源码中见到的. 1.viewport 我见到的时候是这样: <meta name="viewport" content="widt ...
- MyBatis/Ibatis中#和$的区别
1. #将传入的数据都当成一个字符串,会对自动传入的数据加一个双引号.如:order by #user_id#,如果传入的值是111,那么解析成sql时的值为order by "111&qu ...
- Eclipse新建web项目正常启动tomcat不报错,但不能访问项目的解决方法
原因: 虽然我手动添加了自己下载的tomcat,但是由于在Eclipse中创建Server时,"Server Locations"选项采用的时默认配置,即"Use wo ...
- WCF初探-23:WCF中使用Message类(下)
前言 在上一篇WCF中使用Message类(上)中,文章介绍了WCF中使用Message类的基本知识和怎样创建消息,本文是承接上一篇文章,如果想要更好的阅读本文,请先阅读上一篇文章.在这篇文章中,我将 ...