欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 财经 > 创投人物 > 【C++学习笔记 19】C++中的对象生存周期

【C++学习笔记 19】C++中的对象生存周期

2024/10/24 18:22:36 来源:https://blog.csdn.net/m0_59669215/article/details/141906618  浏览:    关键词:【C++学习笔记 19】C++中的对象生存周期

对象如何生存在栈上

在C++中,我们每次进入一个作用域时,我们就是在push栈帧。就像把书堆叠起来,将最新的书放在最上层,在这个作用域上创建变量就像在书中写内容,当作用域结束的时候,就把书拿走,此时每个基于栈的变量就结束了。

举个例子

#include <iostream>
#include <string>class Entity
{
public:Entity(){std::cout << "Created Entity!" << std::endl;}~Entity(){std::cout << "Destroyed Entity" << std::endl;}
};int main() {{Entity e;}std::cin.get();
}

创建一个包含构造函数与析构函数的类,并在一个作用域中创建它,将创建类的实例的语句打上断点,步进。会发现在离开作用域时,就会触发析构函数,表明实例被销毁了,这就是在栈上创建变量。

{Entity* e = new Entity();
}

如果将作用域中的语句改为上述,再次debug步进,会发现直到程序结束,析构函数都没有被触发。
这直观体现了在 栈和堆上创建变量的特点。

如何使用

一个典型的错误案例

int* CreateArray() {int array[50];return array;
}

上述函数乍一看似乎没有什么问题,细看却发现,其返回的是局部变量的指针,当函数运行完毕,栈上的内存空间被回收,指针的对象不再是定义的数组。
如果要实现这样的目的,可以将数组创建在堆上,也可以把这里创建的数组赋值给一个在这个作用域外的变量。

class ScopedPtr
{
private:Entity* m_ptr;
public:ScopedPtr(Entity* ptr): m_ptr(ptr){}~ScopedPtr() {delete m_ptr;}
};

如果想在离开作用域时自动删除堆上创建的变量,可以采用上述的类,一个简单的作用域指针。

{ScopedPtr e(new Entity());
}

并使用它创建实例,由于ScopedPtr的实例是在栈上创建的,因此在离开作用域时会销毁,而此时会调用析构函数,从而删除Entity的指针。这实现了smart ptrunique ptr的基本功能。
这种在离开作用域时指针被销毁的功能很有用,比如说可以做一个作用域的计时器,在离开时自动停止计时。或者可以写一个自动的作用域锁,在函数开始时锁定它,然后在结束时解锁。


教程来源:The Cherno C++ 教程

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com