当前位置: 首页 > news >正文

智能指针(2)

目录

答题格式:

几个易错点

可能问法四:

问题分析:

问题解答:


上回文说到,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;

http://www.lryc.cn/news/456644.html

相关文章:

  • [含文档+PPT+源码等]精品基于Nodejs实现的家教服务小程序的设计与实现
  • electron打包报错-winCodeSign无法下载
  • 给Windows系统设置代理的操作方法
  • 高质量带货短视频素材来源推荐
  • torchvision.transforms.Resize()的用法
  • 简单认识 redis -数据类型命令
  • Python 语言学习——应用1.2 数字图像处理(第二节,变换)
  • 【QT Quick】页面布局:手动定位与坐标系转换
  • uniapp自定义导航,全端兼容
  • [论文阅读] DVQA: Understanding Data Visualizations via Question Answering
  • 【PostgreSQL】实战篇——数据备份和恢复的最佳实践和工具
  • 代码随想录算法训练营第二十九天|93.复原IP地址 78.子集 90.子集II
  • 【mysql】使用AbstractRoutingDataSource实现多数据源 与 获取mapper上注解
  • 希沃冰点还原
  • Hadoop服务端口号、Spark端口号、Hive端口号以及启动命令
  • 【C++】--类和对象(3)
  • 国外电商系统开发-运维系统文件上传-高级上传
  • 【MongoDB】mongodb | 部署 | 常用命令
  • 【Chrome浏览器插件--资源嗅探猫抓】
  • 2.4Mybatis——缓存机制
  • 移动技术开发:文件的读取
  • Linux 中的 Makefile 伪目标详解
  • Java基础(中)
  • Leetcode热题100-200 岛屿数量
  • 大数据新视界 --大数据大厂之 GraphQL 在大数据查询中的创新应用:优化数据获取效率
  • swift使用代码结构解析
  • 五、Python基础语法(程序的输入和输出)
  • 【C语言】常见概念
  • Electron应用创建和打包
  • 代码随想录算法训练营第五六天| 99. 岛屿数量 100. 岛屿的最大面积