博客
关于我
【C++11】C++智能指针循环引用问题分析
阅读量:488 次
发布时间:2019-03-06

本文共 1755 字,大约阅读时间需要 5 分钟。

�_supported_ptr简明指南:避免内存泄漏与循环引用

1. 智能指针的引入与作用

C++11版本的引入三种智能指针:shared_ptr、weak_ptr和unique_ptr,旨在解决动态内存管理中常见的问题。传统的动态内存分配通过new和delete操作可能导致内存泄漏,尤其是在异常处理中抛出错误时。

在传统的内存管理中:

try {    int* p = new int;    // 某些操作后抛出异常    delete p;} catch(...) {// 处理异常}

如果catch块未执行delete操作,会导致内存泄漏。

2. shared_ptr的优势与局限

shared_ptr通过引用计数机制管理动态内存,实现内存管理的简便操作。每个指针可以通过shared_ptr的默认构造来增加引用计数,避免了手动管理内存的危险。

shared_ptr的主要优点是:-自动释放内存:当引用计数为零时,shared_ptr会自动调用析构函数,释放所指的堆内存。-异常安全:确保在异常发生时不会释放内存。

然而,共享指针的局限性在于循环引用时可能导致内存泄漏,例如:

class A;class B;class A {public:    shared_ptr ptr;};class B {public:    shared_ptr ptr;};int main() {    while(true) {        shared_ptr pa(new A());        shared_ptr pb(new B());        pa->ptr = pb;        pb->ptr = pa;    }    return 0;}

在这种情况下,A和B的智能指针循环引用,导致它们的析构函数从未被调用,违反了动态内存释放的条件。

3. weak_ptr:解决循环引用问题的利器

为了应对shared_ptr在循环引用中的局限性,C++引入了weak_ptr。weak_ptr放弃了引用计数机制的功能,通过弱指针存储对象指针,能够在对象生命周期结束时自动释放。

weak_ptr的主要特点:-不管理内存:不会增加引用计数,其他共享指针仍可使用。-不支持操作:不能通过*和->访问对象指针,必须通过unique_ptr配合才能使用。

4. 使用案例分析

考虑如下代码:

#include 
#include
using namespace std;class B {};class A {public: A() { cout << "A's constructor..." << endl; } ~A() { cout << "A's destructor..." << endl; } weak_ptr
weak_b;};class B {public: B() { cout << "B's constructor..." << endl; } ~B() { cout << "B's destructor..." << endl; } weak_ptr weak_a;};int main() { shared_ptr aa = make_shared(); shared_ptr bb = make_shared(); aa->weak_b = bb; bb->weak_a = aa; return 0;}

在该程序中,main函数创建并共享了A和B的对象。weak_ptr不会增加对象的引用计数,断开循环引用,并因此成功调用对象的析构函数,释放内存。

5. 总结

通过分析shared_ptr的局限性及其在循环引用中的表现,我们得出以下结论:

  • shared_ptr在正常情况下有效地管理内存,但在存在循环引用时会导致内存泄漏。
  • weak_ptr提供了解决方案,通过弱指针存储对象指针,避免不必要的引用计数。
  • 在使用智能指针时,应根据实际场景选择适合的类型,以确保内存安全并正确释放资源。
  • 这样的设计帮助开发者更好地管理动态内存,提升代码的稳定性和安全性。

    转载地址:http://hhndz.baihongyu.com/

    你可能感兴趣的文章
    tableviewcell 中使用autolayout自适应高度
    查看>>
    Symbolic Aggregate approXimation(SAX,符号聚合近似)介绍-ChatGPT4o作答
    查看>>
    Orcale表被锁
    查看>>
    svn访问报错500
    查看>>
    sum(a.YYSR) over (partition by a.hy_dm) 不需要像group by那样需要分组函数。方便。
    查看>>
    ORCHARD 是什么?
    查看>>
    Struts2中使用Session的两种方法
    查看>>
    order by rand()
    查看>>
    Orderer节点启动报错解决方案:Not bootstrapping because of 3 existing channels
    查看>>
    org.apache.axis2.AxisFault: org.apache.axis2.databinding.ADBException: Unexpected subelement profile
    查看>>
    org.apache.commons.beanutils.BasicDynaBean cannot be cast to ...
    查看>>
    org.apache.dubbo.common.serialize.SerializationException: com.alibaba.fastjson2.JSONException: not s
    查看>>
    sqlserver学习笔记(三)—— 为数据库添加新的用户
    查看>>
    org.apache.http.conn.HttpHostConnectException: Connection to refused
    查看>>
    org.apache.ibatis.exceptions.PersistenceException:
    查看>>
    org.apache.ibatis.exceptions.TooManyResultsException: Expected one result (or null) to be returned
    查看>>
    org.apache.ibatis.type.TypeException: Could not resolve type alias 'xxxx'异常
    查看>>
    org.apache.poi.hssf.util.Region
    查看>>
    org.apache.xmlbeans.XmlOptions.setEntityExpansionLimit(I)Lorg/apache/xmlbeans/XmlOptions;
    查看>>
    org.apache.zookeeper.KeeperException$ConnectionLossException: KeeperErrorCode = ConnectionLoss for /
    查看>>