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

C++智能指针weak_ptr

weak_ptr 是 C++11 引入的智能指针之一,通常与 shared_ptr 配合使用,用于解决 shared_ptr 可能出现的循环引用问题。


一、什么是 weak_ptr

weak_ptr 是一种“弱引用”智能指针,它不会增加所管理对象的引用计数。它的主要作用是用来观察或访问由 shared_ptr 管理的资源,而不拥有该资源的所有权。由于不拥有所有权,weak_ptr 指向的对象可能在某些情况下已经被销毁,因此在使用时需要特别小心。


二、基本特点

  1. 不控制生命周期weak_ptr 不会影响对象的引用计数,因此不会阻止对象被销毁。
  2. 依赖 shared_ptrweak_ptr 必须从一个 shared_ptr 或另一个 weak_ptr 创建。
  3. 解决循环引用:当两个对象通过 shared_ptr 互相引用时,会导致内存泄漏。使用 weak_ptr 可以打破这种循环。
  4. 需要检查有效性:由于 weak_ptr 指向的对象可能已经被销毁,使用前需要通过 lock() 方法转换为 shared_ptr 并检查是否有效。

三、基本用法

以下是 weak_ptr 的常见用法及其代码示例:

1. 创建 weak_ptr

weak_ptr 通常从 shared_ptr 初始化:

#include <iostream>
#include <memory>

int main() {
    std::shared_ptr<int> sp = std::make_shared<int>(42);
    std::weak_ptr<int> wp = sp; // 从 shared_ptr 创建 weak_ptr

    std::cout << "shared_ptr use_count: " << sp.use_count() << std::endl; // 输出 1
    return 0;
}
  • wp 是从 sp 创建的弱引用。
  • sp.use_count() 返回 1,因为 weak_ptr 不增加引用计数。

2. 使用 lock() 获取 shared_ptr

在使用 weak_ptr 访问对象之前,必须通过 lock() 将其转换为 shared_ptr,并检查是否有效:

#include <iostream>
#include <memory>

int main() {
    std::shared_ptr<int> sp = std::make_shared<int>(42);
    std::weak_ptr<int> wp = sp;

    if (auto locked = wp.lock()) { // 尝试获取 shared_ptr
        std::cout << "Value: " << *locked << std::endl; // 输出 42
        std::cout << "use_count: " << locked.use_count() << std::endl; // 输出 2
    } else {
        std::cout << "Object has been destroyed" << std::endl;
    }

    sp.reset(); // 释放 shared_ptr
    if (auto locked = wp.lock()) {
        std::cout << "Value: " << *locked << std::endl;
    } else {
        std::cout << "Object has been destroyed" << std::endl; // 输出此行
    }

    return 0;
}
  • lock() 返回一个 shared_ptr,如果对象仍存在则有效,否则返回空指针。
  • sp 被销毁后,wp.lock() 返回空指针,表示资源已不可用。

3. 检查是否过期

可以用 expired() 方法检查 weak_ptr 是否指向已销毁的对象:

#include <iostream>
#include <memory>

int main() {
    std::weak_ptr<int> wp;
    {
        std::shared_ptr<int> sp = std::make_shared<int>(42);
        wp = sp;
        std::cout << "Expired? " << wp.expired() << std::endl; // 输出 0 (false)
    }
    std::cout << "Expired? " << wp.expired() << std::endl; // 输出 1 (true)
    return 0;
}
  • expired() 返回 true 表示对象已销毁,false 表示仍有效。

四、解决循环引用问题

weak_ptr 的核心应用场景是打破 shared_ptr 的循环引用。以下是一个经典例子:

循环引用问题

#include <iostream>
#include <memory>

struct Node {
    std::shared_ptr<Node> next;
    ~Node() { std::cout << "Node destroyed" << std::endl; }
};

int main() {
    auto node1 = std::make_shared<Node>();
    auto node2 = std::make_shared<Node>();
    node1->next = node2;
    node2->next = node1; // 循环引用
    std::cout << "End of main" << std::endl;
    return 0;
}
  • 输出只有 “End of main”,Node 的析构函数未被调用,因为 node1node2 互相持有对方的 shared_ptr,引用计数无法降到 0。

使用 weak_ptr 解决问题

#include <memory>
#include <iostream>

struct Node{
    std::weak_ptr<Node> next;
    ~Node(){
        std::cout << "Node deleted" << std::endl;
    }
};

int main() {
    std::weak_ptr<Node> wp1,wp2;
    {
        auto node1 = std::make_shared<Node>();
        auto node2 = std::make_shared<Node>();
        node1->next = node2;
        node2->next = node1;
        wp1 = node1;
        wp2 = node2;
        std::cout << node1.use_count() <<"," << node2.use_count() << std::endl;
    }
    std::cout << wp1.use_count() <<"," << wp2.use_count() << std::endl;

    std::cout << "End of main " << std::endl;
    return 0;
}
  • 输出:
      1,1
      Node deleted
      Node deleted
      0,0
      End of main
    
  • 使用 weak_ptr 后,循环引用被打破,对象能正常销毁。

五、注意事项

  1. 不能直接解引用weak_ptr 没有 operator*operator->,必须通过 lock() 转换为 shared_ptr
  2. 线程安全weak_ptrlock() 操作是线程安全的,但对其本身的赋值或构造不是。
  3. 性能开销weak_ptr 的使用会引入少量开销(如检查有效性),但在需要避免循环引用时是值得的。

六、总结

  • 用途weak_ptr 主要用于观察资源或打破 shared_ptr 的循环引用。
  • 关键方法lock() 获取 shared_ptrexpired() 检查有效性。
  • 典型场景:对象间存在潜在循环引用(如链表、树结构)时。

相关文章:

  • 从0开始的操作系统手搓教程24——完成我们的键盘驱动子系统
  • Ubuntu的软件源
  • 23种设计模式简介
  • 深入理解sync pool
  • 《深入浅出数据索引》- 大参林信息中心内部培训课程笔记
  • CefSharp 文件下载和保存功能-监听前端事件
  • Unity光照之Halo组件
  • 基于Spark的热门动漫推荐数据分析与可视化系统的设计与实现(采用Python语言Django框架,Hadoop,spider爬虫等技术实现)
  • 基于SpringBoot的智慧停车场小程序(源码+论文+部署教程)
  • 【科一】综合素质
  • 【轻松学C:编程小白的大冒险】---常量的定义、声明与应用场景 07
  • 13.数据结构(软考)
  • TypeError: Assignment to constant variable.
  • WebSocket:实现实时通信的利器
  • 【Bert系列模型】
  • 【记录一下学习】Embedding 与向量数据库
  • 结合rpart包的决策树介绍
  • 新疆美女演员吾热叶提正式加入创星演员出道计划,开启演艺事业新篇章
  • 【HeadFirst系列之HeadFirst设计模式】第17天之深入责任链模式:应对复杂请求处理的设计思路
  • 深入解析Seata:分布式事务的终极解决方案
  • 推荐一个可以看片儿的浏览器/seo常用的工具
  • 网站建设太原/网站建设是什么
  • 遵义网约车资格证哪里申请/搜索引擎外部链接优化
  • 进口食品销售销售在那个网站做/搜索seo优化
  • 建设厅质监总站网站/网页代码模板
  • 哈尔滨网站建设培训学校/今日军事新闻最新消息新闻