C++学习第二十八课:C++ 中的智能指针详解

2024-05-11 13:28

本文主要是介绍C++学习第二十八课:C++ 中的智能指针详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

在 C++ 中,内存管理是每个程序员都需要面对的问题。在处理动态分配的内存时,如果忘记释放内存,可能会导致内存泄漏。为了解决这个问题,C++11 引入了智能指针的概念。本文将详细介绍 C++ 中使用智能指针的方法,并结合实际案例进行讲解。

一、什么是智能指针

智能指针是一种对象,它可以像常规指针一样使用,但具有自动管理内存的功能。当智能指针离开其作用域时,它会自动删除所指向的对象,从而避免内存泄漏。C++11 提供了两种智能指针:std::unique_ptrstd::shared_ptr

二、std::unique_ptr 详解

std::unique_ptr 是一种独占所有权的智能指针,即同一时间只能有一个 std::unique_ptr 指向一个对象。当 std::unique_ptr 被销毁时,它会自动删除所指向的对象。

示例代码:

#include <iostream>
#include <memory>class Test {
public:Test() { std::cout << "Test 构造函数" << std::endl; }~Test() { std::cout << "Test 析构函数" << std::endl; }
};int main() {std::unique_ptr<Test> ptr(new Test());// 当 ptr 离开作用域时,自动调用 Test 的析构函数return 0;
}

输出结果:

Test 构造函数
Test 析构函数

三、std::shared_ptr 详解

std::shared_ptr 是一种共享所有权的智能指针,即可以有多个 std::shared_ptr 指向同一个对象。当最后一个 std::shared_ptr 被销毁时,它会自动删除所指向的对象。

示例代码:

#include <iostream>
#include <memory>class Test {
public:Test() { std::cout << "Test 构造函数" << std::endl; }~Test() { std::cout << "Test 析构函数" << std::endl; }
};int main() {{std::shared_ptr<Test> ptr1(new Test());std::shared_ptr<Test> ptr2 = ptr1;// 当 ptr1 和 ptr2 离开作用域时,自动调用 Test 的析构函数}return 0;
}

输出结果:

Test 构造函数
Test 析构函数

四、std::make_uniquestd::make_shared

为了更方便地创建智能指针,C++11 提供了 std::make_uniquestd::make_shared 函数。这两个函数可以简化智能指针的创建过程,并提高代码的可读性。

示例代码:

#include <iostream>
#include <memory>class Test {
public:Test() { std::cout << "Test 构造函数" << std::endl; }~Test() { std::cout << "Test 析构函数" << std::endl; }
};int main() {// 使用 std::make_unique 创建 unique_ptrstd::unique_ptr<Test> ptr1 = std::make_unique<Test>();// 使用 std::make_shared 创建 shared_ptrstd::shared_ptr<Test> ptr2 = std::make_shared<Test>();return 0;
}

输出结果:

Test 构造函数
Test 构造函数

五、智能指针的使用注意事项

  1. 不要使用 delete 操作符删除智能指针所指向的对象,这会导致双重释放问题。
  2. 不要将原始指针(raw pointer)和智能指针混用,这可能导致内存泄漏或双重释放问题。
  3. 不要将两个 std::unique_ptr 赋值给同一个对象,这会导致所有权冲突。
  4. 不要将 std::unique_ptr 转换为 std::shared_ptr,这可能导致循环引用问题。

智能指针是C++中用于自动管理对象生命周期的工具,它们通过引用计数或其他机制来确保在不再需要时自动释放对象。以下是对智能指针类型的细致讲解:

  1. 深复制(Deep Copy)
  • 深复制是指在复制对象时,不仅复制对象的值,还复制对象所指向的内存内容。
  • 对于智能指针而言,深复制意味着创建一个新的动态分配的对象,并将原对象的内容复制到新对象中,然后让智能指针管理这个新的对象。
  • std::shared_ptrstd::unique_ptr 都支持深复制,因为它们都会创建一个新的动态分配的对象。
  1. 写时复制(Copy-on-Write, CoW)
  • 写时复制是一种优化策略,当多个智能指针共享同一个对象时,只有在其中一个智能指针试图修改对象时,才会进行实际的复制操作。
  • std::shared_ptr 使用写时复制机制来提高效率,在没有写入操作时,多个 shared_ptr 可以安全地共享同一个对象。
  1. 引用计数智能指针(Reference Counting Smart Pointers)
  • 引用计数智能指针通过维护一个引用计数来跟踪有多少个智能指针指向同一个对象。
  • std::shared_ptr 是一个引用计数智能指针,当最后一个 shared_ptr 被销毁时,它会自动释放所管理的对象。
  1. 引用链接智能指针(Reference Linking Smart Pointers)
  • 引用链接智能指针通常不直接管理对象的生命周期,而是通过观察其他智能指针来间接实现。
  • std::weak_ptr 是一种引用链接智能指针,它提供了一种方式来避免潜在的循环引用问题,但不会增加对象的引用计数。
  1. 破坏性复制(Disastrous Copy)
  • 破坏性复制并不是智能指针的一个标准概念,但可以理解为在复制过程中可能导致意外行为的情况。
  • 例如,当一个 shared_ptr 管理的资源被另一个 shared_ptr 或原始指针释放时,可能会导致悬挂指针或重复释放资源的问题。

这篇关于C++学习第二十八课:C++ 中的智能指针详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/979663

相关文章

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

mac中资源库在哪? macOS资源库文件夹详解

《mac中资源库在哪?macOS资源库文件夹详解》经常使用Mac电脑的用户会发现,找不到Mac电脑的资源库,我们怎么打开资源库并使用呢?下面我们就来看看macOS资源库文件夹详解... 在 MACOS 系统中,「资源库」文件夹是用来存放操作系统和 App 设置的核心位置。虽然平时我们很少直接跟它打交道,但了

关于Maven中pom.xml文件配置详解

《关于Maven中pom.xml文件配置详解》pom.xml是Maven项目的核心配置文件,它描述了项目的结构、依赖关系、构建配置等信息,通过合理配置pom.xml,可以提高项目的可维护性和构建效率... 目录1. POM文件的基本结构1.1 项目基本信息2. 项目属性2.1 引用属性3. 项目依赖4. 构

Rust 数据类型详解

《Rust数据类型详解》本文介绍了Rust编程语言中的标量类型和复合类型,标量类型包括整数、浮点数、布尔和字符,而复合类型则包括元组和数组,标量类型用于表示单个值,具有不同的表示和范围,本文介绍的非... 目录一、标量类型(Scalar Types)1. 整数类型(Integer Types)1.1 整数字

Java操作ElasticSearch的实例详解

《Java操作ElasticSearch的实例详解》Elasticsearch是一个分布式的搜索和分析引擎,广泛用于全文搜索、日志分析等场景,本文将介绍如何在Java应用中使用Elastics... 目录简介环境准备1. 安装 Elasticsearch2. 添加依赖连接 Elasticsearch1. 创

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

PyTorch使用教程之Tensor包详解

《PyTorch使用教程之Tensor包详解》这篇文章介绍了PyTorch中的张量(Tensor)数据结构,包括张量的数据类型、初始化、常用操作、属性等,张量是PyTorch框架中的核心数据结构,支持... 目录1、张量Tensor2、数据类型3、初始化(构造张量)4、常用操作5、常用属性5.1 存储(st

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

Python 中 requests 与 aiohttp 在实际项目中的选择策略详解

《Python中requests与aiohttp在实际项目中的选择策略详解》本文主要介绍了Python爬虫开发中常用的两个库requests和aiohttp的使用方法及其区别,通过实际项目案... 目录一、requests 库二、aiohttp 库三、requests 和 aiohttp 的比较四、requ

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情