目录
上回文说到,weak_ptr的功能和区别,并且进行了分析,我们接着进行解答。
答题格式:
int main()
{
std::weak_ptr<ListNode> wp;
std::shared_ptr<ListNode> sp;
{
std::shared_ptr<ListNode> n1(new ListNode);
wp = n1;//虽然weak_ptr不支持直接管理资源,但是可以管理shared_ptr已经管理的空间,这边是直接拷贝了一个shared_ptr对象,也可以使用make_shared
//weak_ptr会遇到引用失效的问题, 主要是因为引用对象被置空了,主要还是因为引用计数没有像shared_ptr那样及时更新的问题。然后其weak_ptr自己本身又没有析构函数无法及时释放
cout << wp.expired() << endl;//expired判断会不会引用失效,如果失效了就是真的就会返回1,没有就会返回0
n1->_data++;
sp = wp.lock();//lock的作用是放止引用消失,把wp的资源给跟他同生命周期的sp管理,防止其过期,因为shared_ptr一般不会出现引用失效的问题,所以这个lock就相当于讲weak_ptr提升为类型为shared_ptr的sp,这也说明了weak_ptr可以管理shared_ptr已经管理的资源相当于提升了。
}
cout << wp.expired() << endl;//出了作用域n1销毁了,此时wp就指向了一个空的空间
//就引用失效了,原本会输出1的,但是加上锁lock就不会了
return 0;
}
所以结合上面代码答案如下:
上面提到的expired 和lock面试官没问就不必说了
几个易错点
1。这些智能指针在用的时候还是要加上头文件memory,这里你发现不加也没错主要是因为编译器隐式包含了。具体解释如下:
虽然VS对于某些语法的优化很厉害,但是这里绝对不会优化掉头文件的。所以还是带上比较好。
2。shared_ptr是不会出现引用失效的,因为它的引用计数是会变的,但是这里如果引用的是一个空的对象,也就是说shared_ptr为nullptr,那这里的引用计数就需要特殊处理的,实际是给0的,因为引用空的根本不需要管理。
3。weak_ptr是有引用计数的,属于弱引用,只是引用计数不会随着管理资源的改变而改变,也就是不增加引用计数。
可能问法四:
你知道什么是循环引用吗,怎么解决这种情况?
问题分析:
这个问题就是在考察shared_ptr的缺点已经weak_ptr的研发原因?属于背一背就可以了,最多就是将一个例子。
问题解答:
循环引用自然就是你引用着我,我又引用着你呗,你想释放资源,我引用着你,你得等我释放了才能释放,但是我想释放又被你引用着,这样绕来绕去就随都释放不了了。一下例子可做参考。
struct ListNode
{
int _data;
shared_ptr<ListNode> _next;
shared_ptr<ListNode> _prev;
~ListNode()
{
cout << "~ListNode()" << endl;
}
};
int main()
{
// 循环引用 -- 内存泄露
shared_ptr<ListNode> n1(new ListNode);
shared_ptr<ListNode> n2(new ListNode);
cout << n1.use_count() << endl;
cout << n2.use_count() << endl;
n1->_next = n2;
n2->_prev = n1;
cout << n1.use_count() << endl;
cout << n2.use_count() << endl;
//在循环的地方使用weak_ptr就不会出现这种问题
//weak_ptr不支持管理资源,不支持RAII
return 0;
}
可以看到引用计数正常增加了,结果无法正常析构。
下面解释一下为什么:
结论:照成这种原因的本质就是引用计数的正常增加导致内部循环的部分无法释放,所以解决问题的关键计数使得内部循环引用的部分的引用计数不会增加就可以了,这时weak_ptr不会增加引用计数的优势就体现了,只需要将链表内部循环的部分的管理的智能指针换成weak_ptr就可以了。
//weak_ptr<ListNode> _next;
//weak_ptr<ListNode> _prev;