小心weak_ptr的lock()生成的临时shared_ptr导致reset释放不掉内存

个人通过chatGPT学习weak_ptr,得到如下答案

weak_ptr用法

weak_ptr 是一种智能指针类型,它指向一个 shared_ptr 指向的内存,但不会增加共享指针的引用计数。

因为 weak_ptr 不会增加共享指针的引用计数,所以当共享指针释放内存时,weak_ptr 也会失效,无法通过 weak_ptr 访问内存。

使用 weak_ptr 的方法如下:

引入 weak_ptr 头文件:#include

创建 weak_ptr 实例:std::weak_ptr wptr = ptr;

访问 weak_ptr 指向的内存:std::shared_ptr sptr = wptr.lock();

如果 weak_ptr 无效,则 sptr 的值为 nullptr。

例如:

#include <iostream>
#include <memory>

int main()
{
    // 创建一个指向 int 类型的共享指针
    std::shared_ptr<int> ptr = std::make_shared<int>(10);

    // 创建一个 weak_ptr
    std::weak_ptr<int> wptr = ptr;

    // 访问 weak_ptr 指向的内存
    std::shared_ptr<int> sptr = wptr.lock();
    if (sptr)
    {
        std::cout << *sptr << std::endl;
    }

    // 释放共享指针指向的内存
    ptr.reset();

    // 再次访问 weak_ptr 指向的内存
    sptr = wptr.lock();
    if (!sptr)
    {
        std::cout << "weak_ptr is invalid" << std::endl;
    }

    return 0;
}

运行得不到weak_ptr is invalid的结果

经过自己打印日志发现 lock()生成的临时shared_ptr可能会导致reset()内存释放不掉

具体原因是: shared_ptr<int> sptr = wptr.lock();会新建一个新的对应内存的引用,所以导致内存块不释放

#include <iostream>
#include <unistd.h>
#include <memory>
using namespace std;

int main() {
    // 创建一个指向 int 类型的共享指针
    shared_ptr<int> ptr = make_shared<int>(10);

    // 创建一个 weak_ptr
    weak_ptr<int> wptr = ptr;

    // 访问 weak_ptr 指向的内存
    // 被chatGPT的示例代码坑惨了,这里会新建一个新的对应内存的引用,所以导致内存块不释放
    shared_ptr<int> sptr = wptr.lock();
    if (sptr) {
        cout << *sptr << endl;
    }

     // 输出共享指针的引用计数
    cout << "before reset: " << ptr.use_count() << endl;

    // 释放共享指针指向的内存
    ptr.reset();

    // 输出共享指针的引用计数
    cout << "after reset: " << ptr.use_count() << endl;
    cout << "after reset: sptr count " << sptr.use_count() << endl;
    sptr.reset();
    cout << "after sptr reset: sptr count " << sptr.use_count() << endl;

    cout << "expired ? : " <<  wptr.expired() << endl;
    // 再次访问 weak_ptr 指向的内存
    sptr = wptr.lock();
    cout << "after reset: " << ptr.use_count() << endl;
    if (!sptr) {
        cout << "weak_ptr is invalid" << endl;
    } else {
        cout << sptr << " " << *sptr << endl;
    }
    while(!wptr.expired()) {
        sleep(1);
        cout << "not expired, sleep 1s..." << endl;
    }
    if (!sptr) {
        cout << "weak_ptr is invalid" << endl;
    }

    return 0;
}


/* 
[wolfdan@cloud cpp]$ g++ A.cpp -o A.o -g --std=c++11 
[wolfdan@cloud cpp]$ ./A.o 
10
before reset: 2
after reset: 0
expired ? : 0
after reset: 0
0x13af028 10
not expired, sleep 1s...
not expired, sleep 1s...
not expired, sleep 1s...
^C
[wolfdan@cloud cpp]$ uname -a
Linux cloud 3.10.0-1127.el7.x86_64 #1 SMP Tue Mar 31 23:36:51 UTC 2020 x86_64 x86_64 x86_64 GNU/Linux
[wolfdan@cloud cpp]$ g++ --version
g++ (GCC) 4.8.5 20150623 (Red Hat 4.8.5-39)
Copyright (C) 2015 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.


[wolfdan@cloud cpp]$ g++ A.cpp -o A.o -g --std=c++11 
[wolfdan@cloud cpp]$ ./A.o 
10
before reset: 2
after reset: 0
after reset: sptr count 1
after sptr reset: sptr count 0
expired ? : 1
after reset: 0
weak_ptr is invalid
weak_ptr is invalid
 */
  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值