模板与c++11--右值引用
函数参数传递
struct huge_data{
char *content;
unsigned sz;
huge_data():content(),sz(){
cout<<this<<"->huge_data()"<<endl;
}
huge_data(unsigned sz):content(new char[sz]),sz(sz){
cout<<this<<"->huge_data("<<sz<<")"<<endl;
}
huge_data(huge_data const &h):content(new char[h.sz]),sz(h.sz){
copy(h.content,h.content+sz,content);
cout<<this<<"->huge_data("<<&h<<")"<<endl;
}
huge_data &operator=(huge_data const &h){
if(&h == this) return *this;
if(content) delete[] content;
content = new char[h.sz];
copy(h.content,h.content+sz,content);
sz = h.sz;
cout<<this<<"->operator=("<<&h<<")"<<endl;
return *this;
}
~huge_data(){
if(content){
delete[] content;
}
}
};
huge_data prepare_data(unsigned sz){
huge_data h(sz);
memset(h.content,'a',sizeof(char)*sz);
return h;
}
class BFS{
public:
void test(){
cout<<"begin"<<endl;
huge_data a;
a = prepare_data();
cout<<"end"<<endl;
}
};
参数传递的详细步骤:
1,在test函数的call back上预留能保存huge_data实例的空间,并将其地址作为prepare_data的参数传递到调用栈
2,prepare_data调用huge_data的构造函数为其局部变量h创建huge_data的实例,在函数返回时将h的内容copy构造到test函数的预留空间中
3,test函数再调用赋值操作符=从值从预留空间赋值到a上并释放预留空间
在编译取消优化选项g++ -fno-elide-constructors test.cpp
begin
0x7ffecce0c7f0->huge_data()
0x7ffecce0c7b0->huge_data()
0x7ffecce0c800->huge_data(0x7ffecce0c7b0)
0x7ffecce0c7f0->operator=(0x7ffecce0c800)
end Process returned (0x0) execution time : 0.003 s
Press ENTER to continue.
如果使用编译优化选项的话,就是编译时去掉-fno-elide-constructors
begin
0x7ffd15fa0db0->huge_data()
0x7ffd15fa0dc0->huge_data()
0x7ffd15fa0db0->operator=(0x7ffd15fa0dc0)
end
可以看到省略了一次copy构造函数的调用,
-fno-elide-constructors- The C++ standard allows an implementation to omit creating a temporary which is only used to initialize another object of the same type. Specifying this option disables that optimization, and forces G++ to call the copy constructor in all cases.
g++ 编译选项,no-elide-constructors,标准c++允许这样的机制,当被用来初始化另一个相同类型的另外对象时,省略产生临时变量。如果 指定了-fno-elide-constructors选项,可以禁止此项优化,强制使g++在所有的cases中调用copy constructor。
=========================================
左值,非左值
在c语言中可用于区分赋值和不可赋值的对象。左值可放在复制语句、等号左边的对象,也就是可以进行赋值的对象。常见的左值包括任何变量、指针去引用表达式,而典型的非左值由常数和函数调用。
int *pass(int *p) {return p;}
void main(){
int a,*p;
a = 1;
p = &a;//指针取引用表达式也是左值
1 = 2;//常量1不能进行复制,非左值
pass(p) = &a;//函数返回值也不是左值
*pass(p) = 1;//但是指针去引用后,还是左值
}
c++中左值的本质含义: B.Kernighan与D.Ritchie给出的,所谓对象是agiel具有名称的存储空间,1左值则是意味着某个对象的表达式。
就是说,如果表达式意指一个具名的存储空间,也就意味着随后这一存储空间中的内容还可以依据其名字而获取,则当前对该存储空间的值进行修改(如赋值),才可行。
非左值,1根本没有存储空间的数值,如各种常数,可能在编译时直接优化成指令中的常数而没有存储空间,显然其内容也无法修改。
2指某个无名存储空间的表达式(右值)。
=============
右值,指某个无名存储空间的表达式,如果某个存储空间无名,意味着现在修改了其中的内容,也无法找到该空间再提取其内容,以前的c++标准是禁止修改右值中的内容。
但是在c++11引入右值引用后,是可以的。
某一类型的右值引用类型写为T&&,原有引用类型T&表示左值引用。
====什么情况可以使用右值引用?
右值引用赋予了我们更改右值内容的权利,如左值引用类型一样,右值引用类型可用于函数参数,也可以声明局部变量。
=====移动构造与移动复制
模板与c++11--右值引用的更多相关文章
- [c++11]右值引用、移动语义和完美转发
c++中引入了右值引用和移动语义,可以避免无谓的复制,提高程序性能.有点难理解,于是花时间整理一下自己的理解. 左值.右值 C++中所有的值都必然属于左值.右值二者之一.左值是指表达式结束后依然存在的 ...
- C++11右值引用
[C++11右值引用] 1.什么是左值?什么是右值? 左值是表达式结束后依然存在的对象:右值是表达式结束时就不再存在的对象. 2.std::move的作用是什么? std::move用于把任意类型转化 ...
- 关于C++11右值引用和移动语义的探究
关于C++11右值引用和移动语义的探究
- c++11——右值引用
1. 左值和右值 左值是表达式结束之后仍然存在的持久化对象,而右值是指表达式结束时就不再存在的临时对象. c++11中,右值分为两种类型:将亡值(xvalue, expiring value) ...
- C++ 11 右值引用
C++11中引入的一个非常重要的概念就是右值引用.理解右值引用是学习“移动语义”(move semantics)的基础.而要理解右值引用,就必须先区分左值与右值. 注意:左值右值翻译可能有些问题 *L ...
- C++ 11 右值引用以及std::move
转载请注明出处:http://blog.csdn.net/luotuo44/article/details/46779063 新类型: int和int&是什么?都是类型.int是整数类型,in ...
- 【转】C++ 11 右值引用的理解
右值引用的目的之一,是为了C++中一个比较影响性能的问题:拷贝临时对象,例如,在 int foo(){ ... } int x; x = foo(); 中,在第三句中,发生了以下的事情: 1.销毁 x ...
- C++11右值引用和std::move语句实例解析
关键字:C++11,右值引用,rvalue,std::move,VS 2015 OS:Windows 10 右值引用(及其支持的Move语意和完美转发)是C++0x将要加入的最重大语言特性之一.从实践 ...
- C++11 右值引用和转移语义
新特性的目的 右值引用 (Rvalue Referene) 是 C++ 新标准 (C++11, 11 代表 2011 年 ) 中引入的新特性 , 它实现了转移语义 (Move Sementics) 和 ...
- C++11 右值引用 与 转移语义
新特性的目的 右值引用(R-value Reference)是C++新标准(C++11, 11代表2011年)中引入的新特性,它实现了转移语义(Move Semantics)和精确传递(Perfect ...
随机推荐
- JS整数验证
整数验证 方法1 function ValidatInteger(obj) { var reg = /^[1-9]\d*$/ if (!reg.test($(obj).val())) { $(obj) ...
- weka属性选择使用
醉了--- package edu.dcy.weka; import java.io.FileWriter; import java.util.ArrayList; import java.util. ...
- 【文件拷贝】使用Total Commander Portable拖动拷贝文件,支持队列
使用Total Commander Portable,可以批量拷贝多个位于相同或不同目录的文件(夹)到指定的相同或不同的目录.这样避免了同时复制多个大的文件造成的速度减慢:将所有任务手动操作之后,剩下 ...
- IOS 隐式动画(非Root Layer)
● 每一个UIView内部都默认关联着一个CALayer,我们可用称这个Layer为Root Layer(根 层) ● 所有的非Root Layer,也就是手动创建的CALayer对象,都存在着隐式动 ...
- 前台使用load一个集合后台接受的方法
前台: var imageCaseList = []; }; imageCaseList.push(data); $('#showData').load(url, { querys: imageCas ...
- 前端知识体系之CSS及其预处理器SASS/LESS
如果你是个前端设计师,很多时候我们都在写CSS,CSS是定义页面样式的脚本,并不是一种编程语言,只是一行行单纯的描述页面元素的样子,如果对英语熟练的话,它像说话一样简单,这里举个简单的例子: body ...
- CDQ分治入门
前言 \(CDQ\)分治是一个神奇的算法. 它有着广泛的用途,甚至在某些题目中还能取代\(KD-Tree\).树套树等恶心的数据结构成为正解,而且常数还小得多. 不过它也有一定的缺点,如必须离线操作, ...
- final关键字,static关键字
Final final的意思为最终,不可变.final是个修饰符,它可以用来修饰类,类的成员,以及局部变量.不能修饰构造方法. 注意: 被final修饰的类不能被继承但可以继承别的类 class Yy ...
- C#的接口基础教程之五 实现接口
1.显式实现接口成员 为了实现接口,类可以定义显式接口成员执行体(Explicit interface member implementations).显式接口成员执行体可以是一个方法.一个属性.一个 ...
- react的Redux基础
redux的中文文档:http://www.redux.org.cn/ redux的英文官网:https://redux.js.org/ redux相当于vuex Redux 是 JavaScript ...