【原创】c++拷贝初始化和直接初始化的底层区别
源代码
#include <iostream>
#include <cstring>
using namespace std;
class ClassTest
{
public:
ClassTest()
{
c[] = '\0';
cout << "ClassTest()" << endl;
}
ClassTest& operator=(const ClassTest &ct)
{
strcpy(c, ct.c);
cout << "ClassTest& operator=(const ClassTest &ct)" << endl;
return *this;
}
ClassTest(ClassTest&& ct)
{
cout << "ClassTest(ClassTest&& ct)" << endl;
}
ClassTest & operator=(ClassTest&& ct)
{
strcpy(c, ct.c);
cout << "ClassTest & operator=(ClassTest&& ct)" << endl;
return *this;
}
ClassTest(const char *pc)
{
strcpy(c, pc);
cout << "ClassTest (const char *pc)" << endl;
}
//private:
ClassTest(const ClassTest& ct)
{
strcpy(c, ct.c);
cout << "ClassTest(const ClassTest& ct)" << endl;
}
virtual int ff()
{
return ;
}
private:
char c[];
};
ClassTest f1()
{
ClassTest c;
return c;
}
void f2(ClassTest ct)
{
;
}
int main()
{
ClassTest ct1("ab");//直接初始化
ClassTest ct2 = "ab";//复制初始化
ClassTest ct3 = ct1;//复制初始化
ClassTest ct4(ct1);//直接初始化
ClassTest ct5 = ClassTest("ab");//复制初始化
ClassTest ct6 = f1();
f1();
f2(ct1);
return ;
}
初始化1:ClassTest ct1("ab")
ClassTest ct1("ab");//直接初始化
00B09518 push 0B0DCB8h //"ab"字符串地址
00B0951D lea ecx,[ct1]
00B09523 call ClassTest::ClassTest (0DC101Eh)
初始化2:ClassTest ct2 = "ab"
ClassTest ct2 = "ab";//复制初始化
00B09528 push 0B0DCB8h //"ab"字符串地址
00B0952D lea ecx,[ct2]
00B09533 call ClassTest::ClassTest (0DC101Eh)
初始化3:ClassTest ct3 = ct1
ClassTest ct3 = ct1;//复制初始化
00B09538 lea eax,[ct1]
00B0953E push eax
00B0953F lea ecx,[ct3]
00B09545 call ClassTest::ClassTest (0DC14C4h)
010B3EE0 push ebp
010B3EE1 mov ebp,esp
010B3EE3 sub esp,0CCh
010B3EE9 push ebx
010B3EEA push esi
010B3EEB push edi
010B3EEC push ecx
010B3EED lea edi,[ebp-0CCh]
010B3EF3 mov ecx,33h
010B3EF8 mov eax,0CCCCCCCCh
010B3EFD rep stos dword ptr es:[edi]
010B3EFF pop ecx
010B3F00 mov dword ptr [this],ecx
010B3F03 mov eax,dword ptr [this] //eax指向ct3对象地址
010B3F06 mov dword ptr [eax],10BDC70h //虚表指针存储在对象偏移量为0的地方
010B3F0C mov esi,dword ptr [__that] //esi存储ct1对象地址
010B3F0F add esi, //将esi加4,跳过4个字节的虚表指针,指向ct1后面的成员变量c
010B3F12 mov edi,dword ptr [this]
010B3F15 add edi, //edi指向ct2后面成员变量c
010B3F18 mov ecx,40h
010B3F1D rep movs dword ptr es:[edi],dword ptr [esi] //将ct1中字符数组元素拷贝到ct3字符数组
010B3F1F mov eax,dword ptr [this] //通过eax返回ct3对象地址
010B3F22 pop edi
010B3F23 pop esi
010B3F24 pop ebx
010B3F25 mov esp,ebp
010B3F27 pop ebp
初始化4:ClassTest ct4(ct1)
ClassTest ct4(ct1);//直接初始化
010B954A lea eax,[ct1]
010B9550 push eax
010B9551 lea ecx,[ct4]
010B9557 call ClassTest::ClassTest (0DC14C4h)
初始化5:ClassTest ct5 = ClassTest()
ClassTest ct5 = ClassTest();//复制初始化
010B955C lea ecx,[ct5]
010B9562 call ClassTest::ClassTest (0DC12ADh)
ClassTest()
010B4C70 push ebp
010B4C71 mov ebp,esp
010B4C73 sub esp,0CCh
010B4C79 push ebx
010B4C7A push esi
010B4C7B push edi
010B4C7C push ecx
010B4C7D lea edi,[ebp-0CCh]
010B4C83 mov ecx,33h
010B4C88 mov eax,0CCCCCCCCh
010B4C8D rep stos dword ptr es:[edi]
010B4C8F pop ecx
010B4C90 mov dword ptr [this],ecx
010B4C93 mov eax,dword ptr [this]
010B4C96 mov dword ptr [eax],10BDC70h
{
c[] = '\0';
010B4C9C mov eax,
010B4CA1 imul ecx,eax,
010B4CA4 mov edx,dword ptr [this]
010B4CA7 mov byte ptr [edx+ecx+],
cout << "ClassTest()" << endl;
初始化6:ClassTest ct6 = f1()
ClassTest ct6 = f1();
010B9567 lea eax,[ct6]
010B956D push eax
010B956E call f1 (0DC14BFh)
010B9573 add esp,
ClassTest f1()
{
00DC5830 push ebp //栈帧开始
00DC5831 mov ebp,esp
00DC5833 sub esp,1D0h
00DC5839 push ebx
00DC583A push esi
00DC583B push edi
00DC583C lea edi,[ebp-1D0h]
00DC5842 mov ecx,74h
00DC5847 mov eax,0CCCCCCCCh
00DC584C rep stos dword ptr es:[edi]
00DC584E mov eax,dword ptr ds:[00DD0000h] //初始化栈
00DC5853 xor eax,ebp
00DC5855 mov dword ptr [ebp-],eax
ClassTest c;
00DC5858 lea ecx,[c] //c的值ebp+FFFFFEF4h即ebp-,说明c是一个栈内局部变量
00DC585E call ClassTest::ClassTest (0DC12ADh) //调用默认构造函数初始化c
return c;
00DC5863 lea eax,[c]
00DC5869 push eax //c对象地址
00DC586A mov ecx,dword ptr [ebp+] //ct6对象地址
00DC586D call ClassTest::ClassTest (0DC14BAh) //调用移动构造函数,初始化ct6
00DC5872 mov eax,dword ptr [ebp+] //返回ct6对象地址
}
00DC5875 push edx
00DC5876 mov ecx,ebp
00DC5878 push eax
00DC5879 lea edx,ds:[0DC58A4h]
00DC587F call @_RTC_CheckStackVars@8 (0DC1136h)
00DC5884 pop eax
//省略余下代码
临时对象:f1()
f1();
00DC9576 lea eax,[ebp-814h]
00DC957C push eax
00DC957D call f1 (0DC14BFh)
00DC9582 add esp,
临时对象:f2(ct1)
f2(ct1);
010F9392 sub esp,104h //开辟栈空间,生成一个临时对象,刚好是260个字节(+,即虚表指针和私有的char型数组的总大小)
010F9398 mov ecx,esp //将esp栈顶指针作为临时对象的起始地址
010F939A lea eax,[ct1] //传入ct1对象地址
010F93A0 push eax
010F93A1 call ClassTest::ClassTest (010F1078h)
010F93A6 call f2 (010F14BFh)
010F93AB add esp,104h
总结
- 使用赋值运算符定义变量
- 将对象作为实参传递给一个非引用类型的形参
- 将一个返回类型为非引用类型的函数返回一个对象
- 用花括号列表初始化一个数组中的元素或一个聚合类中的成员
ClassTest ct2 ="ab"; //相当于ClassTest ct2("ab");
ClassTest ct5 =ClassTest("ab"); //相当于ClassTest ct5("ab")
下面的语句,visual studio才会生成一个无名的临时对象(位于main函数的栈中),注意:f1的返回值类型是非引用的,f2的形参类型是非引用的。
f1(); //临时对象用于存储f1的返回值
f2(ct1); //临时对象用于拷贝实参,并传入函数
而下面则是直接传入赋值表达式左边对象地址,然后再对该对象进行移动拷贝,注意f1返回值类型是非引用的,如果是引用的,则会调用拷贝构造函数。
ClassTest ct6 = f1();
- 将一个返回类型为非引用类型的函数返回一个对象
- 赋值表达式右边是一个对象
- 直接初始化时,括号内的参数是一个对象
- 用花括号列表初始化一个数组中的元素或一个聚合类中的成员
- 将一个返回类型为引用类型的函数返回一个对象
- 形参为非引用类型的函数,其中是将实参拷贝到临时对象
- 赋值表达式右边是一个左值对象(如果需要,可以调用构造函数类型转换,生成一个临时对象)
- 当赋值表达式右边是一个右值对象,且没有定义移动赋值运算符函数
- 当赋值表达式右边是一个右值对象,且定义了移动赋值运算符函数
ClassTest ct2 = "ab";//复制初始化
ClassTest ct2("ab");//直接初始化
本文链接:【原创】c++拷贝初始化和直接初始化的底层区别 http://www.cnblogs.com/cposture/p/4925736.html
【原创】c++拷贝初始化和直接初始化的底层区别的更多相关文章
- 【c++】必须在类初始化列表中初始化的几种情况
转自:http://www.cnblogs.com/kaituorensheng/p/3477630.html 1. 类成员为const类型 2. 类成员为引用类型 #include <iost ...
- 代码初始化 故事板初始化 xib初始化总结
对象的初始化有三种方式 // 代码创建 - (id)initWithFrame:(CGRect)frame { if (self = [super initWithFrame:frame]) { ...
- c++构造函数成员初始化中赋值和初始化列表两种方式的区别
先总结下: 由于类成员初始化总在构造函数执行之前 1)从必要性: a. 成员是类或结构,且构造函数带参数:成员初始化时无法调用缺省(无参)构造函数 b. 成员是常量或引用:成员无法赋值,只能被初始化 ...
- Spark源码剖析 - SparkContext的初始化(八)_初始化管理器BlockManager
8.初始化管理器BlockManager 无论是Spark的初始化阶段还是任务提交.执行阶段,始终离不开存储体系.Spark为了避免Hadoop读写磁盘的I/O操作成为性能瓶颈,优先将配置信息.计算结 ...
- Java静态初始化,实例初始化以及构造方法
首先有三个概念需要了解: 一.静态初始化:是指执行静态初始化块里面的内容. 二.实例初始化:是指执行实例初始化块里面的内容. 三.构造方法:一个名称跟类的名称一样的方法,特殊在于不带返回值. 我们先来 ...
- C#中结构(struct)的部分初始化和完全初始化
假设有这样一个值类型struct. public struct Size { public int Length; public int Width; public int Area() { retu ...
- java类的成员初始化顺序和初始化块知识
java类的成员初始化顺序和初始化块知识 转自:http://blog.csdn.net/lgfeng218/article/details/7606735 属性.方法.构造方法和自由块都是类中的成员 ...
- C++的一大误区——深入解释直接初始化与复制初始化的区别
转自:http://blog.csdn.net/ljianhui/article/details/9245661 不久前,在博客上发表了一篇文章——提高程序运行效率的10个简单方法,对于其中最后一 ...
- C++直接初始化和复制初始化1
这篇文章主要介绍了C++直接初始化与复制初始化的区别深入解析,是很多C++初学者需要深入了解的重要概念,需要的朋友可以参考下 C++中直接初始化与复制初始化是很多初学者容易混淆的概念,本文就以实例 ...
随机推荐
- HDU 6346 整数规划 (最佳完美匹配)
整数规划 Time Limit: 5500/5000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total Subm ...
- Lucene用法示例
整理一下 ELK 和 Grafana 中会用到的 Lucene 用法: 通配符 示例1:过滤出 url 中包含 .pw/ 的 网址 url.keyword:*.pw\/* 正则表达式 示例1:过滤出 ...
- Apache启动不成功时,用命令行检测(新手)
1,在配置Apache服务器时,经常要在httpd.conf 修改和添加一些代码,编写中,误写或者写错时,无法正常启动时,直接报错The requested operation has failed! ...
- java笔试之输出
1. public class foo { private static void testMethod(){ System.out.println("testMethod"); ...
- JVM运行时数据区(一)
1.名词解释: 栈帧:栈帧是用于支持虚拟机进行方法调用和方法执行的数据结构,它是虚拟机运行时数据区中的虚拟机栈的栈元素. 2.程序计数器: 程序计数器是一块比较小的内存空间,可以将它看作是当前线程所执 ...
- 201671010147 2017年8月27号 初学java的感想
在IT行业中,java无疑是最热门的,很多企业也青睐java,因为他的扩展性好,可以处理更多客户的数据,正是因为java有前景所以才吸引更多人去学习.在大一我们已经接触vhleC语言,大二开始就解除了 ...
- 在码云(gitee)上展开程序类课程教学
码云主要提供了源代码管理(Git/SVN)功能,最近又推出了高校版让普通老师也能利用起来以供教学使用. 学生与老师不仅能利用其管理代码,更重要的是我们的程序教学能通过对git的使用来引入业界流行的软件 ...
- 【.NET Core项目实战-统一认证平台】第二章网关篇-定制Ocelot来满足需求
[.NET Core项目实战-统一认证平台]开篇及目录索引 这篇文章,我们将从Ocelot的中间件源码分析,目前Ocelot已经实现那些功能,还有那些功能在我们实际项目中暂时还未实现,如果我们要使用这 ...
- 【webpack】-- 自动刷新与解析
前端需要频繁的修改js和样式,且需要根据浏览器的页面效果不断的做调整:而且往往我们的开发目录和本地发布目录不是同一个,修改之后需要发布一下:另外一点就是并不是所有的效果都可以直接双击页面就能看到,我们 ...
- 背水一战 Windows 10 (117) - 后台任务: 后台下载任务
[源码下载] 背水一战 Windows 10 (117) - 后台任务: 后台下载任务 作者:webabcd 介绍背水一战 Windows 10 之 后台任务 后台下载任务 示例演示 uwp 的后台下 ...