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

智能指针讲解

 写重载取地址运算符发现的好例子,干脆单独写一篇来讲

1. 什么是智能指针?

智能指针是 C++ 标准库提供的一种类模板,它封装了原始指针并提供了自动管理内存的功能,避免了手动管理内存(newdelete)时容易出现的错误,如内存泄漏和野指针等问题。

2. std::unique_ptr

std::unique_ptr 是一种 独占所有权 的智能指针,它表示一个指针拥有一个资源的所有权,并且资源的所有权不能被共享。 std::unique_ptr 在其生命周期结束时会自动释放资源。

特点
  • 不能被复制:unique_ptr 只能通过移动(std::move)来转移所有权。
  • 自动释放资源:当 unique_ptr 被销毁时,它会自动释放其指向的资源。
#include <iostream>
#include <memory> // 包含 unique_ptr

class MyClass {
public:
    MyClass() { std::cout << "MyClass Constructor!" << std::endl; }
    ~MyClass() { std::cout << "MyClass Destructor!" << std::endl; }

    void display() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};

int main() {
    // 使用 unique_ptr 管理 MyClass 对象
    std::unique_ptr<MyClass> ptr1 = std::make_unique<MyClass>();
    
    // 调用成员函数
    ptr1->display();
    
    // 在这里 ptr1 被销毁,自动调用析构函数,释放 MyClass 对象
    return 0;
}
  • std::unique_ptr<MyClass> ptr1 = std::make_unique<MyClass>();std::make_unique 用来创建一个 unique_ptr 并初始化它,ptr1 拥有这个 MyClass 对象的所有权。
  • ptr1 离开作用域时,它会自动释放内存,调用 MyClass 的析构函数。

 

你可以看到,unique_ptr 保证了 MyClass 对象在不再使用时会被自动销毁。

注意:
  • 不能复制 unique_ptr,只能转移所有权。反例如下面代码
std::unique_ptr<MyClass> ptr2 = ptr1; // 编译错误,不能复制 unique_ptr

不过你可以使用 std::move 转移所有权:

std::unique_ptr<MyClass> ptr2 = std::move(ptr1); // 正确,ptr1 的所有权被转移给 ptr2

3. std::shared_ptr

std::shared_ptr 是一种 共享所有权 的智能指针,它允许多个 shared_ptr 对象共同拥有同一个资源。资源会被自动释放,当所有指向资源的 shared_ptr 被销毁时,资源才会被释放。

特点
  • 共享所有权:多个 shared_ptr 可以指向同一个资源。
  • 引用计数shared_ptr 内部维护一个引用计数,记录有多少个 shared_ptr 指向同一个对象。当引用计数为 0 时,资源才会被销毁。
  • 线程不安全:引用计数的增加和减少不是线程安全的,如果需要跨线程使用,需要自行加锁。
#include <iostream>
#include <memory> // 包含 shared_ptr

class MyClass {
public:
    MyClass() { std::cout << "MyClass Constructor!" << std::endl; }
    ~MyClass() { std::cout << "MyClass Destructor!" << std::endl; }

    void display() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};

int main() {
    // 使用 shared_ptr 管理 MyClass 对象
    std::shared_ptr<MyClass> ptr1 = std::make_shared<MyClass>();

    // 共享 ptr1 的所有权
    std::shared_ptr<MyClass> ptr2 = ptr1;
    
    std::cout << "Reference count: " << ptr1.use_count() << std::endl; // 输出 2

    // 调用成员函数
    ptr1->display();
    ptr2->display();
    
    std::cout << "Reference count: " << ptr1.use_count() << std::endl; // 输出 2

    return 0;
}
  • std::shared_ptr<MyClass> ptr1 = std::make_shared<MyClass>();:创建一个 shared_ptr,它管理一个 MyClass 对象。
  • std::shared_ptr<MyClass> ptr2 = ptr1;ptr2 共享了 ptr1 所指向的 MyClass 对象,两者都有对该对象的所有权。
  • ptr1.use_count() 返回引用计数,表示当前有多少个 shared_ptr 指向同一个对象。

 

可以看到,当多个 shared_ptr 共享对同一个资源的所有权时,引用计数在 use_count() 调用时显示为 2。当 shared_ptr 对象 ptr1ptr2 都超出作用域时,MyClass 对象的析构函数会被调用。

4. std::weak_ptr

std::weak_ptr 是一种弱引用智能指针,通常与 std::shared_ptr 配合使用。它不参与引用计数的增加或减少,避免了循环引用问题。weak_ptr 用于观察一个 shared_ptr 管理的对象,但不拥有它。

#include <iostream>
#include <memory>

class MyClass {
public:
    MyClass() { std::cout << "MyClass Constructor!" << std::endl; }
    ~MyClass() { std::cout << "MyClass Destructor!" << std::endl; }
    void display() { std::cout << "Hello from MyClass!" << std::endl; }
};

int main() {
    std::shared_ptr<MyClass> ptr1 = std::make_shared<MyClass>();
    std::weak_ptr<MyClass> weakPtr = ptr1; // weak_ptr 观察 ptr1
    
    std::cout << "Reference count: " << ptr1.use_count() << std::endl; // 输出 1

    if (auto sharedPtr = weakPtr.lock()) { // lock() 返回一个 shared_ptr
        sharedPtr->display(); // 如果 weak_ptr 指向的对象仍然存在,使用它
    } else {
        std::cout << "Object no longer exists!" << std::endl;
    }

    return 0;
}
  • std::weak_ptr<MyClass> weakPtr = ptr1;weakPtr 是对 ptr1 管理的对象的弱引用,不增加引用计数。
  • weakPtr.lock()lock() 尝试将 weak_ptr 转换为 shared_ptr。如果对象已经被销毁,lock() 返回 nullptr,否则返回一个有效的 shared_ptr

做个总结吧 

  • std::unique_ptr:表示独占所有权,不能复制,只能通过 std::move 转移所有权,自动管理资源。
  • std::shared_ptr:表示共享所有权,多个 shared_ptr 可以共同拥有一个资源,引用计数机制保证资源的自动释放。
  • std::weak_ptr:不增加引用计数,避免循环引用问题,用于观察和访问由 shared_ptr 管理的资源。

智能指针提供了安全、自动管理内存的方式,帮助避免了手动管理内存时可能出现的错误,如内存泄漏、双重释放等。尤其是在现代 C++ 编程中,智能指针是一个非常强大的工具,能够大大简化内存管理。

相关文章:

  • 深度学习与人工智能资料推荐:DeepSeek整合包全解析
  • 智享AI直播三代系统,马斯克旗下AI人工智能直播工具,媲美DeepSeek!
  • 大摩闭门会:250218 学习总结报告
  • 排序算法:冒泡排序
  • ARM64 Trust Firmware [四]
  • 【鸿蒙开发】第三十八章 ArkTS代码调试
  • Spotify AI 技术(1)使用 TensorFlow 和 TF-Agents
  • 电脑机箱散热风扇声音大的影响因素
  • Linux I2C 命令
  • 最小二乘法原理分析
  • C语言指针学习笔记
  • 深入浅出TypedArray:网络数据处理、WebGPU与加密实战
  • 从入门到跑路(六)k8s配置ingress-nginx
  • 代码随想录算法【Day50】
  • Pycharm中查找与替换
  • 计算机网络-OSI七层参考模型与数据封装,网络安全零基础入门到精通实战教程!
  • ZLMediaKit Windows 编译指南
  • 大模型WebUI:Gradio全解11——使用transformers.agents构建Gradio UI(3)
  • Java-如何将其他地方拉取的jar包导入本地maven环境
  • vue3中<el-table-column>状态的显示
  • 男子恶意遗弃幼子获刑,最高法发布涉未成年人家庭保护典型案例
  • 冰雹造成车损能赔吗?如何理赔?机构答疑
  • 违法违规收集使用个人信息,爱奇艺、轻颜等65款App被点名
  • 广东省中医院脾胃病科大科主任张北平病逝,年仅52岁
  • 复旦大学与上海杨浦共建市东医院
  • “救护车”半路加价?陕西卫健委已介入,记者调查:黑救护车挤占市场