个人通过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
*/