热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

请大神帮忙看下这个程序

#include<iostream>#include<stack>usingnamespacestd;classA{public:A(intelem
#include 
#include 
using namespace std;

class A
{
public:
A(int element)
{
a = element;
cout<<"construct"< }
~A()
{
cout<<"end"< }
void print()
{
cout< }
private:
int a;
};

int main()
{
A a1(1),a2(2);
stack stack1;
stack1.push(a1);
stack1.push(a2);
A &data = stack1.top();
stack1.pop();
data.print();
return 0;
}


有两个问题:
1、构造两个对象,构造函数调用2次,析构函数为什么会调用4次呢?
2、stack1.pop()后,a2对象被析构了,那为什么它的引用data还能够调用成员函数print()呢?析构后对象内存不应该释放才对嘛?
本人c++新手,还请大神们帮忙解答一下,谢谢!

10 个解决方案

#1


1. 有两个对象是通过调用A类的复制构造函数生成的。

2. 
a、 此时a2还没有析构,它的作用域是main函数,在main函数结束时才会调用它的析构函数
b、 push 操作会复制对象,引用的并不是a2,而是stack中复制生成的副本

#2


引用 1 楼 mougaidong 的回复:
1. 有两个对象是通过调用A类的复制构造函数生成的。

2. 
a、 此时a2还没有析构,它的作用域是main函数,在main函数结束时才会调用它的析构函数
b、 push 操作会复制对象,引用的并不是a2,而是stack中复制生成的副本

恩,感谢您的回复,也就是说push时调用了类A的复制构造函数,其push只是原对象的一个副本。
不过还有个问题,data引用的是a2的副本,暂且叫它a2_temp,那pop()之后a2_temp会被析构才对,那为什么data.print()还能够执行呢

#3


push的时候传的是引用,没有发生复制啊,奇怪了

#4


http://bbs.csdn.net/topics/390472811

引用 2 楼 xc889078 的回复:
Quote: 引用 1 楼 mougaidong 的回复:

1. 有两个对象是通过调用A类的复制构造函数生成的。

2. 
a、 此时a2还没有析构,它的作用域是main函数,在main函数结束时才会调用它的析构函数
b、 push 操作会复制对象,引用的并不是a2,而是stack中复制生成的副本

恩,感谢您的回复,也就是说push时调用了类A的复制构造函数,其push只是原对象的一个副本。
不过还有个问题,data引用的是a2的副本,暂且叫它a2_temp,那pop()之后a2_temp会被析构才对,那为什么data.print()还能够执行呢

#5


传的是引用不假,但是,你有看过push的实现吗?

引用 3 楼 henry3695 的回复:
push的时候传的是引用,没有发生复制啊,奇怪了

#7


1、两个push其实是构造了两个对象
2、pop后可以访问是因为,对象析构后,只是告诉系统此对象已经无效了,但是数据的内存值还在 的,等到下次用到这块内存的时候,直接把这块内存使用了,这样做貌似好偷懒,其实是提高了效率,少了一次内存操作,只是个人推测

#8



A test(A TT)
{
TT.print();
return TT;
}


在加点,看看调用这个要做几次拷贝

#9


引用 8 楼 henry3695 的回复:

A test(A TT)
{
TT.print();
return TT;
}


在加点,看看调用这个要做几次拷贝

这应该是调用两次复制拷贝,一次是形参值传递那里,另一次是return那里,所以会执行两次复制拷贝函数和两次析构函数

#10


计算机组成原理→DOS命令→汇编语言→C语言(不包括C++)、代码书写规范→数据结构、编译原理、操作系统→计算机网络、数据库原理、正则表达式→其它语言(包括C++)、架构……

对学习编程者的忠告:
眼过千遍不如手过一遍!
书看千行不如手敲一行!
手敲千行不如单步一行!
单步源代码千行不如单步对应汇编一行!

单步类的实例“构造”或“复制”或“作为函数参数”或“作为函数返回值返回”或“参加各种运算”或“退出作用域”的语句对应的汇编代码几步后,就会来到该类的“构造函数”或“复制构造函数”或“运算符重载”或“析构函数”对应的C/C++源代码处。

VC调试时按Alt+8、Alt+7、Alt+6和Alt+5,打开汇编窗口、堆栈窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应堆栈、内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。

推荐阅读
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 本文探讨了 C++ 中普通数组和标准库类型 vector 的初始化方法。普通数组具有固定长度,而 vector 是一种可扩展的容器,允许动态调整大小。文章详细介绍了不同初始化方式及其应用场景,并提供了代码示例以加深理解。 ... [详细]
  • 本文详细介绍了C语言中链表的两种动态创建方法——头插法和尾插法,包括具体的实现代码和运行示例。通过这些内容,读者可以更好地理解和掌握链表的基本操作。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • C++实现经典排序算法
    本文详细介绍了七种经典的排序算法及其性能分析。每种算法的平均、最坏和最好情况的时间复杂度、辅助空间需求以及稳定性都被列出,帮助读者全面了解这些排序方法的特点。 ... [详细]
  • 本文将介绍由密歇根大学Charles Severance教授主讲的顶级Python入门系列课程,该课程广受好评,被誉为Python学习的最佳选择。通过生动有趣的教学方式,帮助初学者轻松掌握编程基础。 ... [详细]
  • Splay Tree 区间操作优化
    本文详细介绍了使用Splay Tree进行区间操作的实现方法,包括插入、删除、修改、翻转和求和等操作。通过这些操作,可以高效地处理动态序列问题,并且代码实现具有一定的挑战性,有助于编程能力的提升。 ... [详细]
  • 本实验主要探讨了二叉排序树(BST)的基本操作,包括创建、查找和删除节点。通过具体实例和代码实现,详细介绍了如何使用递归和非递归方法进行关键字查找,并展示了删除特定节点后的树结构变化。 ... [详细]
  • 文件描述符、文件句柄与打开文件之间的关联解析
    本文详细探讨了文件描述符、文件句柄和打开文件之间的关系,通过具体示例解释了它们在操作系统中的作用及其相互影响。 ... [详细]
  • 本文详细探讨了VxWorks操作系统中双向链表和环形缓冲区的实现原理及使用方法,通过具体示例代码加深理解。 ... [详细]
  • 本教程涵盖OpenGL基础操作及直线光栅化技术,包括点的绘制、简单图形绘制、直线绘制以及DDA和中点画线算法。通过逐步实践,帮助读者掌握OpenGL的基本使用方法。 ... [详细]
  • 本题涉及一棵由N个节点组成的树(共有N-1条边),初始时所有节点均为白色。题目要求处理两种操作:一是改变某个节点的颜色(从白变黑或从黑变白);二是查询从根节点到指定节点路径上的第一个黑色节点,若无则输出-1。 ... [详细]
  • 探索1000以内的完美数:因数和等于自身
    本文探讨了如何在1000以内找到所有完美数,即一个数的因数(不包括自身)之和等于该数本身。例如,6是一个完美数,因为1 + 2 + 3 = 6。通过编程实现这一过程,可以更好地理解完美数的特性。 ... [详细]
author-avatar
Pri_颖颖_773
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有