c++智能指针weak_ptr

1.为什么需要weak_ptr

为了解决shared_ptr循环引用导致内存泄漏的问题,才引入了weak_ptr。通过引入weak_ptr,顺利解决了这个问题。

2.回顾shared_ptr循环引用问题

2.1代码及运行输出

struct TestB;
struct TestA{
    TestA(){
        a = QUuid::createUuid().toString();
        qInfo() << a <<"已经被构造了"<<endl;
    };

    TestA(QString aa){
        a = aa;
        qInfo() << a <<"已经被构造了"<<endl;
    }

    ~TestA(){
        qInfo() << a <<"已经被析构了"<<endl;
    }

    QString a;
    shared_ptr<TestB> b_ptr;
};

struct TestB{
    TestB(){
        b = QUuid::createUuid().toString();
        qInfo() << b <<"已经被构造了"<<endl;
    };

    TestB(QString aa){
        b = aa;
        qInfo() << b <<"已经被构造了"<<endl;
    }

    ~TestB(){
        qInfo() << b <<"已经被析构了"<<endl;
    }

    QString b;
    shared_ptr<TestA> a_ptr;
};

void Weak_ptr::whyNeedWeakPtr()
{
    shared_ptr<TestA> a = make_shared<TestA>();
    shared_ptr<TestB> b = make_shared<TestB>();

    qInfo() << "a的引用计数为:"<<a.use_count()<<endl;
    qInfo() << "b的引用计数为:"<<b.use_count()<<endl;

    a->b_ptr = b;
    b->a_ptr = a;

    qInfo() << "a的引用计数为:"<<a.use_count()<<endl;
    qInfo() << "b的引用计数为:"<<b.use_count()<<endl;
}

输出结果

2.2原因分析

由输出结果可知,在循环引用之后,a,b,两个智能指针的引用计数都变成了2.在出了函数作用域之后,a,b指针的引用计数自动减一,也就是说都变成了1,但不会变为0,也就导致了这两个智能指针指向的对象没有被析构,导致了内存泄漏。
如果想解决这个问题,我们只要想个办法,在这种循环引用的情景下,其中任何一个的引用计数为1(就假比如说a)。这样的话,出函数作用域后,a智能指针的引用计数就变成0,指向对象就会被析构,析构的时候,对象内的b指针应用计数也会变为1.然后b指针也会出作用域,引用计数再次减一,指向的对象也就会正常析构了。
为了达到这个目的,大佬们引入了weak_ptr这个指针。

3.代码改造

我们只需要将上面的TestA中得shared_ptr改为weak_ptr就可以了,改造后的代码如下。

struct TestB;
struct TestA{
    TestA(){
        a = QUuid::createUuid().toString();
        qInfo() << a <<"已经被构造了"<<endl;
    };

    TestA(QString aa){
        a = aa;
        qInfo() << a <<"已经被构造了"<<endl;
    }

    ~TestA(){
        qInfo() << a <<"已经被析构了"<<endl;
    }

    QString a;
    weak_ptr<TestB> b_ptr;
};

然后再次执行之前的whyNeedWeakPtr()函数。输出结果如下
输出结果
从输出图我们就可以看出和我们之前分析的一样,a中使用的weak_ptr,导致对应的引用计数没有加一,也就可以正常的析构了。

4.如何使用weak_ptr

void Weak_ptr::howToUse()
{
    shared_ptr<Test> a = make_shared<Test>();
    qInfo() << "a的引用计数为:" << a.use_count() ;
    weak_ptr<Test> w = a;
    //weak_ptr 的use_count()返回的时 与该weak_ptr共指向的shared_ptr的引用计数
    qInfo() << "w的引用计数为:" <<  w.use_count() << "a的引用计数为:" << a.use_count() << endl;

    shared_ptr<Test> b = w.lock(); //调用lock后如果 指向的内存没有被释放就返回共指向的shared_ptr,并使
    //共指向的shared_ptr的引用计数加一,如果已经被释放就返回空
    qInfo() << "w的引用计数为:" <<  w.use_count() << "a的引用计数为:" << a.use_count() << endl;
    qInfo() << b->getString() << endl;

    //weak_ptr 提供expired判断改指针是否已经失效,
    qInfo() <<"w是否失效"<< w.expired();
    b.reset();
    a.reset();
    qInfo() << "w是否失效" << w.expired();

    shared_ptr<Test> c = make_shared<Test>();
    weak_ptr<Test> w2 = c;
    shared_ptr<Test> d = make_shared<Test>();
    weak_ptr<Test> w3 = d;

    qInfo() << "w2指向值为:" << w2.lock()->getString() << "w3指向值为:" << w3.lock()->getString();
    w2.swap(w3); //交换共指向的shared_ptr
    qInfo() << "w2指向值为:" << w2.lock()->getString() << "w3指向值为:" << w3.lock()->getString(); 
}

输出结果如下

在这里插入图片描述

5.使用weak_ptr注意事项

weak_ptr是一种弱引用,常常用来侦查对象是否存在,不控制对象的生命期,一般作为一个保护防止调用时共指向的shared_ptr已经失效,使用方法如下。

void Weak_ptr::checkIsValid()
{
    shared_ptr<Test> a = make_shared<Test>();
    qInfo() << "a指向值为:" << a->getString() << endl;
    weak_ptr<Test> w(a);
    if(!w.expired()){
        qInfo()<< __LINE__ << w.lock()->getString() << endl;
    }

    a.reset();

    qInfo() << w.expired() <<endl;
    if(!w.expired()){
        qInfo() << __LINE__ << w.lock()->getString() << endl;
    }
}

输出如下
在这里插入图片描述
可以看出只执行了一次输出,在reset之后,w失效,下面的代码没有被执行,有效的进行了保护

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
shared_ptrC++中的智能指针,它采用引用计数的方法来实现释放指针所指向的资源。当使用shared_ptr时,它会记录有多少个shared_ptr指向同一个对象,只有当最后一个shared_ptr被销毁时,该对象的内存才会被释放。因此,shared_ptr可以自动管理内存,不需要手动释放。 在代码中,使用shared_ptr可以像普通指针一样操作对象。当需要创建一个shared_ptr对象时,可以使用std::make_shared函数来构造,如下所示: ``` std::shared_ptr<int> sharedPtr1 = std::make_shared<int>(10); ``` 可以通过将shared_ptr赋值给其他shared_ptr来共享资源,如下所示: ``` std::shared_ptr<int> sharedPtr2 = sharedPtr1; ``` 当所有的shared_ptr都被销毁时,内存会自动释放。可以使用use_count()函数获取shared_ptr的引用计数,如下所示: ``` std::cout << "sharedPtr2 use count: " << sharedPtr2.use_count() << std::endl; ``` 当引用计数为0时,内存会被自动释放。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C++智能指针shared_ptr分析](https://download.csdn.net/download/weixin_38705004/13788082)[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: 50%"] - *2* *3* [C++11中的智能指针unique_ptr、shared_ptrweak_ptr详解](https://blog.csdn.net/chenlycly/article/details/130918547)[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: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值