0

0

C++如何使用智能指针管理动态分配对象

P粉602998670

P粉602998670

发布时间:2025-09-09 12:24:02

|

837人浏览过

|

来源于php中文网

原创

智能指针通过自动管理内存防止泄漏,主要包括shared_ptr、unique_ptr和weak_ptr;shared_ptr用于共享所有权并自动释放资源,但需避免循环引用;unique_ptr确保独占所有权,支持所有权转移但不可复制;weak_ptr用于打破循环引用,观察shared_ptr管理的对象;在性能敏感、与C互操作或嵌入式场景中可考虑原始指针;还可通过自定义删除器管理特殊资源如文件句柄。

c++如何使用智能指针管理动态分配对象

智能指针本质上是为了更安全地管理C++中的动态内存分配。它们通过自动释放不再需要的内存来防止内存泄漏,并简化了资源管理。

shared_ptr、unique_ptr和weak_ptr是C++标准库提供的三种主要智能指针类型,每种类型都有不同的用途和所有权模型。

shared_ptr:允许多个指针指向同一个对象,并使用引用计数来跟踪有多少个shared_ptr指向该对象。当最后一个shared_ptr被销毁时,对象会被自动删除。 unique_ptr:提供对对象的独占所有权。一次只能有一个unique_ptr指向一个对象,当unique_ptr被销毁时,对象会被自动删除。 weak_ptr:不增加对象的引用计数,而是提供对shared_ptr所管理对象的非拥有访问。weak_ptr可以用来检测对象是否仍然存在。

shared_ptr 的使用场景和注意事项

shared_ptr适用于多个对象需要共享同一个资源的情况。例如,多个对象需要访问同一个数据库连接或同一个文件句柄。

#include 
#include 

class MyClass {
public:
    MyClass(int value) : value_(value) {
        std::cout << "MyClass constructor, value: " << value_ << std::endl;
    }
    ~MyClass() {
        std::cout << "MyClass destructor, value: " << value_ << std::endl;
    }

    int getValue() const { return value_; }

private:
    int value_;
};

int main() {
    // 创建一个 shared_ptr 指向 MyClass 对象
    std::shared_ptr ptr1 = std::make_shared(10);
    std::cout << "ptr1 use_count: " << ptr1.use_count() << std::endl; // 输出 1

    // 复制 shared_ptr
    std::shared_ptr ptr2 = ptr1;
    std::cout << "ptr1 use_count: " << ptr1.use_count() << std::endl; // 输出 2
    std::cout << "ptr2 use_count: " << ptr2.use_count() << std::endl; // 输出 2

    // ptr1 和 ptr2 指向同一个对象
    std::cout << "ptr1 value: " << ptr1->getValue() << std::endl;
    std::cout << "ptr2 value: " << ptr2->getValue() << std::endl;

    // 当 ptr1 和 ptr2 都超出作用域时,MyClass 对象才会被销毁
    return 0;
}

注意事项:

立即学习C++免费学习笔记(深入)”;

  • 循环引用: 避免shared_ptr的循环引用,否则会导致内存泄漏。例如,A对象包含一个指向B对象的shared_ptr,而B对象又包含一个指向A对象的shared_ptr。在这种情况下,即使A和B对象不再被使用,它们的引用计数也永远不会降为0,导致内存泄漏。可以使用weak_ptr来打破循环引用。
  • 线程安全: shared_ptr的引用计数是线程安全的,但是shared_ptr所管理的对象本身不一定是线程安全的。如果多个线程同时访问shared_ptr所管理的对象,需要进行适当的同步。
  • make_shared
    vs
    new
    :
    优先使用
    std::make_shared
    创建
    shared_ptr
    make_shared
    可以一次性分配对象和控制块(用于引用计数),从而提高效率并避免潜在的异常安全问题。

unique_ptr 的所有权转移和使用限制

unique_ptr适用于需要确保只有一个指针指向对象的情况。例如,表示文件句柄或网络连接等独占资源。

#include 
#include 

class Resource {
public:
    Resource(std::string name) : name_(name) {
        std::cout << "Resource " << name_ << " acquired." << std::endl;
    }
    ~Resource() {
        std::cout << "Resource " << name_ << " released." << std::endl;
    }

    std::string getName() const { return name_; }

private:
    std::string name_;
};

int main() {
    // 创建一个 unique_ptr 指向 Resource 对象
    std::unique_ptr ptr1 = std::make_unique("File.txt");

    // 所有权转移:ptr1 的所有权转移到 ptr2
    std::unique_ptr ptr2 = std::move(ptr1);

    // ptr1 现在为空
    if (ptr1 == nullptr) {
        std::cout << "ptr1 is now null." << std::endl;
    }

    // ptr2 拥有 Resource 对象
    std::cout << "ptr2 owns resource: " << ptr2->getName() << std::endl;

    // 当 ptr2 超出作用域时,Resource 对象会被自动释放
    return 0;
}

所有权转移:

  • unique_ptr可以通过
    std::move
    函数将所有权转移给另一个unique_ptr。转移后,原来的unique_ptr将变为null。

使用限制:

SlidesAI
SlidesAI

使用SlidesAI的AI在几秒钟内创建演示文稿幻灯片

下载
  • 不能复制unique_ptr。这是因为unique_ptr的设计目标是确保只有一个指针指向对象。如果允许复制unique_ptr,就会违反这个原则。
  • unique_ptr可以用于指向数组。可以使用
    std::unique_ptr
    来管理动态分配的数组。

weak_ptr 如何解决循环引用问题

weak_ptr用于观察shared_ptr所管理的对象,但不增加对象的引用计数。它可以用来检测对象是否仍然存在,并避免循环引用。

#include 
#include 

class B; // 前向声明

class A {
public:
    std::shared_ptr b_ptr;
    ~A() { std::cout << "A is destroyed" << std::endl; }
};

class B {
public:
    std::weak_ptr a_ptr; // 使用 weak_ptr 打破循环引用
    ~B() { std::cout << "B is destroyed" << std::endl; }
};

int main() {
    std::shared_ptr a = std::make_shared();
    std::shared_ptr b = std::make_shared();

    a->b_ptr = b;
    b->a_ptr = a; // b 观察 a,但不拥有 a

    // 当 a 和 b 超出作用域时,它们都会被销毁,不会发生内存泄漏
    return 0;
}

如何解决循环引用:

  • 在一个类中使用weak_ptr指向另一个类,从而打破循环引用。例如,A对象包含一个指向B对象的shared_ptr,而B对象包含一个指向A对象的weak_ptr。在这种情况下,当A和B对象不再被使用时,它们的引用计数最终会降为0,从而避免内存泄漏。
  • 在使用weak_ptr之前,需要先使用
    lock()
    方法将其转换为shared_ptr。如果对象已经被销毁,
    lock()
    方法将返回一个空的shared_ptr。

何时应该避免使用智能指针

尽管智能指针提供了许多好处,但在某些情况下,使用原始指针可能更合适。

  • 性能关键型代码: 智能指针会带来一些性能开销,例如引用计数的维护和内存分配。在性能至关重要的代码中,可以考虑使用原始指针来避免这些开销。但是,需要仔细管理原始指针的生命周期,以避免内存泄漏和悬挂指针。
  • 与C代码的互操作: C代码通常不使用智能指针。如果需要将C++代码与C代码进行互操作,可能需要使用原始指针。
  • 嵌入式系统: 在资源受限的嵌入式系统中,智能指针的内存开销可能是一个问题。在这种情况下,可以使用原始指针或自定义的内存管理方案。

如何自定义删除器来管理特殊资源

智能指针可以使用自定义删除器来管理特殊资源,例如文件句柄或网络连接。

#include 
#include 
#include 

// 自定义删除器,用于关闭文件
struct FileDeleter {
    void operator()(std::ofstream* file) {
        if (file->is_open()) {
            file->close();
            std::cout << "File closed." << std::endl;
        }
        delete file;
    }
};

int main() {
    // 创建一个 shared_ptr,并使用自定义删除器
    std::shared_ptr file(new std::ofstream("example.txt"), FileDeleter());

    // 检查文件是否成功打开
    if (file->is_open()) {
        *file << "Hello, world!" << std::endl;
    } else {
        std::cerr << "Unable to open file." << std::endl;
    }

    // 当 file 超出作用域时,FileDeleter 会被调用,文件会被关闭
    return 0;
}

如何使用自定义删除器:

  • 可以向智能指针的构造函数传递一个自定义删除器。删除器可以是函数对象、lambda表达式或函数指针。
  • 当智能指针被销毁时,删除器会被调用,用于释放智能指针所管理的资源。

智能指针是C++中管理动态分配对象的重要工具。理解不同类型智能指针的特性和使用场景,可以帮助你编写更安全、更高效的代码。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

438

2024.03.01

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

443

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

544

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

73

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

207

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

191

2025.11.08

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7.7万人学习

C 教程
C 教程

共75课时 | 4.2万人学习

C++教程
C++教程

共115课时 | 14.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号