每日C++ - shared_ptr 的循环引用问题

shared_ptr 的循环引用问题

在 C++ 开发过程中,面对浩繁的代码工作,不免会出现各种 bug,而在智能指针使用方面也存在一些难以察觉的坑。shared_ptr 循环引用造成内存泄漏就是一例。当两个内存对象分别拥有对方的非 nullptr 的 shared_ptr 类型的智能指针,导致 shared_ptr 引用计数无法归 0,从而造成系统内存泄漏。

我们来看下面的代码

class B;
class A
{
public:
    A() = default;
    ~A() { std::cout << "A dtor." << std::endl;}
    std::shared_ptr<B> b;
};
class B
{
public:
    B() = default;
    ~B() { std::cout << "B dtor." << std::endl; }
    std::shared_ptr<A> a;
};

上面就是典型的错误场景,在实际运行过程中,我们无法看到 A,B 的析构。

int main()
{
	std::shared_ptr<A> sa = std::make_shared<A>();
	std::shared_ptr<B> sb = std::make_shared<B>();
	sa->b = sb;
	sb->a = sa;
	return 0;
}

对于 shared_ptr 循环引用的问题,怎么来解决呢?

  • 在实际代码编写过程中,尽量使用 unique_ptr,避免出现循环引用的场景

    unique_ptr 只能独占使用,我们可以通过右值或裸指针将 unique_ptr 的内存对象传递给其它内存对象。

    class B;
    class A
    {
    public:
    	std::unique_ptr<B> b;
    };
    class B
    {
    public:
    	void test() {}
    };
    int main()
    {
    	std::unique_ptr<A> a = std::make_unique<A>();
    	std::unique_ptr<B> b = std::make_unique<B>();
    	a->b = std::move(b);
    	a->b->test();
    	// b->get() is nullptr
    }
    
  • 如果无法避免循环引用,则改用 weak_ptr

    weak_ptr 并不会增加内存对象的引用计数,为避免内存泄漏,循环引用使用 weak_ptr

    class B;
    class A
    {
    public:
        A() = default;
        ~A() { std::cout << "A dtor." << std::endl; }
        void test() { std::cout << "A test." << std::endl; }
        std::weak_ptr<B> b;
    };
    class B
    {
    public:
         B() = default;
         ~B() { std::cout << "B dtor." << std::endl; }
         void test() { std::cout << "B test." << std::endl; }
         std::weak_ptr<A> a;
    };
    
    我们无法通过 weak_ptr 直接调用内存对象的成员和方法,可以通过 weak_ptr 的 lock() 方法获得它的 shared_ptr,然后再调用内存对象的成员和方法。通过 weak_ptr 的 expired() 方法可以判断它所指向的内存对象是否已被释放。
    int main()
    {
          std::shared_ptr<WeakPtr::A> sa = std::make_shared<WeakPtr::A>();
          std::shared_ptr<WeakPtr::B> sb = std::make_shared<WeakPtr::B>();
          sa->b = sb;
          sb->a = sa;
    
          // shared_ptr call test
          sa->test();
          sb->test();
          // weak_ptr call test
          sa->b.lock()->test();
          sb->a.lock()->test();
    
          // released
          std::cout << "B expired = " << sa->b.expired() << std::endl;
          std::cout << "A expired = " << sb->a.expired() << std::endl;
    }
    
    上述代码运行后,A,B 都会被正常析构。
shared_ptr循环引用是指多个shared_ptr实例互相引用,导致对象无法被释放的情况。shared_ptr采用引用计数的智能指针,可以指向同一个动态对象,并维护了一个共享的引用计数器。当多个shared_ptr实例相互引用时,它们的引用计数无法降为零,从而导致内存泄漏。 为了解决shared_ptr引起的循环引用问题,可以使用weak_ptr指针。weak_ptr是一种弱引用,不能单独使用,只能配合shared_ptr使用。相比之下,weak_ptr并不增加引用计数,它只是提供了对被共享对象的一个非拥有的引用。通过使用weak_ptr,可以打破shared_ptr之间的循环引用,使对象能够正常释放。 在C++中,我们可以使用weak_ptr的lock()函数来获取一个可用的shared_ptr对象,从而访问被共享对象的成员函数和成员变量。weak_ptr没有重载*和->运算符,因此无法直接访问对象,但可以使用lock()函数来获取shared_ptr,并通过该shared_ptr来访问对象的成员。 通过使用weak_ptr解决shared_ptr循环引用问题,可以避免内存泄漏和资源占用过多的情况。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C++11智能指针之weak_ptr详解](https://download.csdn.net/download/weixin_38740201/14841441)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [C++ 智能指针的循环引用问题](https://blog.csdn.net/qq_28584889/article/details/88726324)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [C++智能指针(share_ptr)及其循环引用问题](https://blog.csdn.net/feikudai8460/article/details/104747490)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值