c++ 智能指针实战分析

news2024/12/23 14:43:18

一.智能指针的设计思路

  1. 智能指针是类模板,再栈上创建智能指针对象。
  2. 把普通指针交给智能指针对象。
  3. 智能指针对象过期时,调用析构函数释放普通指针的内存。
  • 智能指针的类型
  1. auto_ptr是C++98的标准,c++17已经弃用。
  2. unique_ptr、shared_ptr和weak_ptr是C++11标准的。

二.智能指针 unique_ptr

C++智能指针unique_ptr是C++11标准库中提供的一种智能指针类型,用于管理动态分配的资源,主要用来避免内存泄漏和简化资源管理。unique_ptr是独占所有权的智能指针,即同一时间只能有一个unique_ptr指向特定的资源,当unique_ptr被销毁时,它所管理的资源会被自动释放。

1.基本用法

  1. 方法1:
unique_ptr<AA>p0(new AA("小明"));//分配内存并初始化
  1. 方法2:
unique_ptr<AA>p0=make_unique<AA>("小明");//c++14标准
  1. 方法3:
AA*p=new AA("小明");
unique_ptr<AA>p0(p);//用已存在的地址初始化
  1. 方法4:get()方法可以返回裸指针

2. 基本定义

  • 包含头文件
#include <memory>

以下是unique_ptr的简化版本的类模板定义(省略了部分实现细节):

template <typename T>
class unique_ptr {
private:
    T* ptr;

public:
    // 构造函数
    unique_ptr(T* p) : ptr(p) {}

    // 移动构造函数
    unique_ptr(unique_ptr&& other) noexcept : ptr(other.ptr) {
        other.ptr = nullptr;
    }

    // 析构函数
    ~unique_ptr() {
        delete ptr;
    }

    // 禁止拷贝和赋值
    unique_ptr(const unique_ptr&) = delete;
    unique_ptr& operator=(const unique_ptr&) = delete;

    // 移动赋值运算符
    unique_ptr& operator=(unique_ptr&& other) noexcept {
        if (this != &other) {
            delete ptr;
            ptr = other.ptr;
            other.ptr = nullptr;
        }
        return *this;
    }

    // 获取裸指针
    T* get() const {
        return ptr;
    }

    // 重载运算符->
    T* operator->() const {
        return ptr;
    }

    // 重载解引用运算符*
    T& operator*() const {
        return *ptr;
    }

    // 释放资源
    void reset() {
        delete ptr;
        ptr = nullptr;
    }
};

3.实战测试

#include<iostream>
using namespace std;

#include<vector>
#include<memory>//智能指针头文件

class AA
{
public:
	string m_name;
	AA():m_name("未定义") { cout << m_name << "调用构造函数AA()" << endl; }

	AA(const string& name) :m_name(name) { cout << "调用构造函数(" << m_name << ")" << endl; }
	~AA() { cout << "调用了析构函数~AA(" << m_name << ")" << endl; }


};


int main()
{
	AA* p1 = new AA("小明");

	//使用智能指针来管理p

	//这里AA的意思是要管理的普通指针类型是AA
	//智能指针本来就是用来管理指针的所以不需要使用AA*
	unique_ptr<AA>ptr1(p1);

	//与正常指针用法一致
	cout << "m_name " << (*ptr1).m_name << endl;
	cout << "m_name " << ptr1->m_name << endl;
	cout << "m_name " << (*p1).m_name << endl;
	cout << "m_name " << p1->m_name << endl;

	//system("pause");不能使用这个,否则程序无法正常关闭,析构函数无法成功调用
	return 0;
}
  • 执行结果

在这里插入图片描述

三.智能指针 shared_ptr

在 C++ 中,智能指针是一种用于管理动态分配对象的指针类,可以帮助开发人员更好地管理内存资源,避免内存泄漏等问题。std::shared_ptr 是 C++11 标准库中提供的一种智能指针,用于共享所有权的指针管理。std::shared_ptr 可以自动追踪有多少个指针共享同一个对象,并在对象不再被引用时安全地释放内存。共享指针的引用计数会在创建、复制和销毁时进行递增和递减,当引用计数为 0 时,内存资源就会被释放。

使用方法:(和unique_ptr基本一样的)

#include <memory>
std::shared_ptr<int> sharedPtr = std::make_shared<int>(10);
智能指针的复制:
std::shared_ptr<int> sharedPtr2 = sharedPtr;
使用 std::make_shared 创建对象时,可以减少内存分配和额外的虚拟函数调用。

可以使用 .use_count() 方法获取当前共享指针的引用计数。

std::shared_ptr 还支持自定义的删除器,用于指定释放内存时的操作。

总的来说,std::shared_ptr 可以帮助避免潜在的内存泄漏和悬空指针问题,提高代码的健壮性和可维护性。

  • 实战测试
#include<iostream>
using namespace std;

#include<vector>
#include<memory>//智能指针头文件

class AA
{
public:
	string m_name;
	AA() :m_name("未定义") { cout << m_name << "调用构造函数AA()" << endl; }

	AA(const string& name) :m_name(name) { cout << "调用构造函数(" << m_name << ")" << endl; }
	~AA() { cout << "调用了析构函数~AA(" << m_name << ")" << endl; }


};


int main()
{
	AA* p1 = new AA("小明");

	//使用智能指针来管理p

	//这里AA的意思是要管理的普通指针类型是AA
	//智能指针本来就是用来管理指针的所以不需要使用AA*
	shared_ptr<AA>ptr1(p1);

	//与正常指针用法一致
	cout << "m_name " << (*ptr1).m_name << endl;
	cout << "m_name " << ptr1->m_name << endl;
	cout << "m_name " << (*p1).m_name << endl;
	cout << "m_name " << p1->m_name << endl;
	cout << "use_count=" << ptr1.use_count() << endl;

	//system("pause");不能使用这个,否则程序无法正常关闭,析构函数无法成功调用
	return 0;
}

  • 输出结果
    在这里插入图片描述

四.智能指针 weak_ptr

在 C++ 中,std::weak_ptr 是另一种智能指针,用于解决 std::shared_ptr 循环引用所带来的问题。std::weak_ptr 允许你观察指向的对象,但不拥有其所有权。通常与 std::shared_ptr 一起使用,可以避免循环引用导致的内存泄漏问题。

以下是一些关于 std::weak_ptr 的重要点和用法:

  1. 通过 std::weak_ptr 对象获std::shared_ptr 对象:
std::shared_ptr<int> sharedPtr = std::make_shared<int>(10);
std::weak_ptr<int> weakPtr = sharedPtr;
  1. 使用 std::lock_guard 和 std::shared_ptr::lock 方法来安全地获取 std::shared_ptr 对象,避免访问已经释放的对象:
if (std::shared_ptr<int> sharedPtr = weakPtr.lock()) {
    // 使用 sharedPtr 访问对象
} else {
    // 对象已被释放
}
  1. std::weak_ptr 不增加引用计数,只是提供了对对象的观察能力,也不会阻止对象的释放。

  2. 可以使用 std::weak_ptr 来打破循环引用,让对象能够被正确地释放。

std::weak_ptr 是一种很有用的工具,可以帮助解决在使用智能指针时可能会遇到的循环引用问题,确保内存资源能够正确地释放,提高程序的稳定性和可靠性。

  • 实战测试
#include<iostream>
using namespace std;

#include<vector>
#include<memory>//智能指针头文件

class BB;

class AA
{
public:
	string m_name;
	AA() :m_name("未定义") { cout << m_name << "调用构造函数AA()" << endl; }

	AA(const string& name) :m_name(name) { cout << "调用构造函数(" << m_name << ")" << endl; }
	~AA() { cout << "调用了析构函数~AA(" << m_name << ")" << endl; }

	shared_ptr<BB>p_B;

};

class BB
{
public:
	string m_name;
	BB() :m_name("未定义") { cout << m_name << "调用构造函数BB()" << endl; }

	BB(const string& name) :m_name(name) { cout << "调用构造函数(" << m_name << ")" << endl; }
	~BB() { cout << "调用了析构函数~BB(" << m_name << ")" << endl; }

	shared_ptr<AA>p_A;
};


int main()
{
	AA* p1 = new AA("小明");
	BB* p2 = new BB("李华");

	//使用智能指针来管理p

	//这里AA的意思是要管理的普通指针类型是AA
	//智能指针本来就是用来管理指针的所以不需要使用AA*
	shared_ptr<AA>ptr1(p1);
	shared_ptr<BB>ptr2 (p2);

	//与正常指针用法一致
	cout << "ptr1->m_name " << (*ptr1).m_name << endl;
	cout << "ptr1->m_name " << ptr1->m_name << endl;
	cout << "ptr1->m_name " << (*p1).m_name << endl;
	cout << "ptr1->m_name " << p1->m_name << endl;
	cout << "ptr1->use_count=" << ptr1.use_count() << endl;

	cout << "ptr2->m_name " << (*ptr2).m_name << endl;
	cout << "ptr2->m_name " << ptr2->m_name << endl;
	cout << "ptr2->m_name " << (*p2).m_name << endl;
	cout << "ptr2->m_name " << p2->m_name << endl;
	cout << "ptr2->use_count=" << ptr2.use_count() << endl;

	ptr1->p_B = ptr2;
	ptr2->p_A = ptr1;

	//system("pause");不能使用这个,否则程序无法正常关闭,析构函数无法成功调用
	return 0;
}
  • 由于相互执行都在等待对方释放资源,结果两方都无法释放资源
    在这里插入图片描述

  • 解决方法:使用weak_ptr

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1883688.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

动手学深度学习(Pytorch版)代码实践 -计算机视觉-41目标检测数据集

41目标检测数据集 import os import pandas as pd import torch import torchvision import matplotlib.pylab as plt from d2l import torch as d2l# 数据集下载链接 # http://d2l-data.s3-accelerate.amazonaws.com/banana-detection.zip# 读取数据集 #save def read_data_b…

Mustango——音乐领域知识生成模型探索

Mustango&#xff1a;利用领域知识的音乐生成模型 论文地址&#xff1a;https://arxiv.org/pdf/2311.08355.pdf 源码地址&#xff1a;https://github.com/amaai-lab/mustango 论文题为**“**利用音乐领域知识开发文本到音乐模型’Mustango’”。它利用音乐领域的知识从文本指…

计算机毕业设计Python深度学习美食推荐系统 美食可视化 美食数据分析大屏 美食爬虫 美团爬虫 机器学习 大数据毕业设计 Django Vue.js

Python美食推荐系统开题报告 一、项目背景与意义 随着互联网和移动技术的飞速发展&#xff0c;人们的生活方式发生了巨大变化&#xff0c;尤其是餐饮行业。在线美食平台如雨后春笋般涌现&#xff0c;为用户提供了丰富的美食选择。然而&#xff0c;如何在海量的餐饮信息中快速…

python 笔试面试八股(自用版~)

1 解释型和编译型语言的区别 解释是翻译一句执行一句&#xff0c;更灵活&#xff0c;eg&#xff1a;python; 解释成机器能理解的指令&#xff0c;而不是二进制码 编译是整个源程序编译成机器可以直接执行的二进制可运行的程序&#xff0c;再运行这个程序 比如c 2 简述下 Pyth…

2.2章节python的变量和常量

在Python中&#xff0c;变量和常量有一些基本的概念和用法&#xff0c;但需要注意的是&#xff0c;Python本身并没有内置的“常量”类型。然而&#xff0c;程序员通常会遵循一种约定&#xff0c;即使用全部大写的变量名来表示常量。 一、变量 在Python中&#xff0c;变量是一…

对不起,AI大模型不是风口

“我们正处在全新起点&#xff0c;这是一个以大模型为核心的人工智能新时代&#xff0c;大模型改变了人工智能&#xff0c;大模型即将改变世界。”——5月26日&#xff0c;百度创始人、董事长兼CEO李彦宏先生在2023中关村论坛发表了《大模型改变世界》演讲。 李彦宏指出&#…

4PCS点云配准算法实现

4PCS点云配准算法的C实现如下&#xff1a; #include <iostream> #include <pcl/io/pcd_io.h> #include <pcl/point_types.h> #include <pcl/common/common.h> #include <pcl/common/distances.h> #include <pcl/common/transforms.h> #in…

PostgreSQL介绍与安装

一、PostgreSQL数据库介绍 1、什么是数据库&#xff1f; 数据库&#xff08;Database&#xff09;是按照数据结构来组织、存储和管理数据的仓库。每个数据库都有一个或多个不同的 API 用于创建&#xff0c;访问&#xff0c;管理&#xff0c;搜索和复制所保存的数据。 我们也…

JAVA医院绩效考核管理系统源码:系统优势、系统目的、系统原则 (自主研发 功能完善 可直接上项目)

JAVA医院绩效考核管理系统源码&#xff1a;系统优势、系统目的、系统原则 &#xff08;自主研发 功能完善 可直接上项目&#xff09; 医院绩效考核系统优势 1.实现科室负责人单独考核 对科室负责人可以进行单独考核、奖金发放。 2. 科室奖金支持发放到个人 支持奖金二次分配&…

同一个excel表格,为什么在有的电脑上会显示#NAME?

一、哪些情况会产生#NAME?的报错 1.公式名称拼写错误 比如求和函数SUM&#xff0c;如果写成SUN就会提示#NAME&#xff1f;报错。 2.公式中的文本值未添加双引号 如下图&#xff1a; VLOOKUP(丙,A:B,2,0) 公式的计算结果会返回错误值#NAME?&#xff0c;这是因为公式中文本…

GraphPad Prism生物医学数据分析软件下载安装 GraphPad Prism轻松绘制各种图表

Prism软件作为一款功能强大的生物医学数据分析与可视化工具&#xff0c;其绘图功能尤为突出。该软件不仅支持绘制基础的图表类型&#xff0c;如直观明了的柱状图、展示数据分布的散点图&#xff0c;以及描绘变化趋势的曲线图&#xff0c;更能应对复杂的数据呈现需求&#xff0c…

7-1作业

1.实验目的&#xff1a;完成字符收发 led.h #ifndef __GPIO_H__ #define __GPIO_H__#include "stm32mp1xx_rcc.h" #include "stm32mp1xx_gpio.h" #include "stm32mp1xx_uart.h"//RCC,GPIO,UART初始化 void init();//字符数据发送 void set_tt…

【2024 插件开发大赛】为 ONLYOFFICE 开发插件,赢取万元奖金!

我们发布了 2024 插件开发大赛&#xff1a;为 ONLYOFFICE 开发适合中国用户的插件&#xff0c;赢取税前5500 – 10000元的结项奖金与证书&#xff01;阅读本文了解详情。 关于 ONLYOFFICE ONLYOFFICE 是一个国际开源项目&#xff0c;由领先的 IT 公司 Ascensio System SIA 开发…

论坛万能粘贴手(可将任意文件转为文本)

该软件可将任意文件转为文本。 还原为原文件的方法&#xff1a;将得到的文本粘贴到记事本&#xff0c;另存为UUE格式&#xff0c;再用压缩软件如winrar解压即可得到原文件。建议用于小软件。 下载地址&#xff1a;https://download.csdn.net/download/wgxds/89505015 使用演示…

算法基础入门 - 2.栈、队列、链表

文章目录 算法基础入门第二章 栈、队列、链表2.1 队列2.2 栈2.3 纸牌游戏2.4 链表如何建立链表?1.我们需要一个头指针(head)指向链表的初始。链表还没建立时头指针head为空2.建立第一个结点3.设置刚创建的这个结点的数据域(左半)和指针域(右半)4.设置头指针,头指针可方便…

构建高效的数字风控系统:应对现代网络威胁的策略与实践

文章目录 构建高效的数字风控系统&#xff1a;应对现代网络威胁的策略与实践1. 数字风控基本概念1.1 数字风控&#xff08;数字化风控&#xff09;1.2 数字风控的原理1.3 常见应用场景 2. 数字风控的必要性3. 构建高效的数字风控系统3.1 顶层设计与规划3.2 数据基础建设3.3 风险…

代码随想录第38天|动态规划

1049. 最后一块石头的重量 II 参考 备注: 当物体容量也等同于价值时, 01背包问题的含义则是利用好最大的背包容量sum/2, 使得结果尽可能的接近或者小于 sum/2 等价: 尽可能的平分成相同的两堆, 其差则为结果, 比如 (abc)-d, (ac)-(bd) , 最终的结果是一堆减去另外一堆的和, 问…

Nik Collection by DxO:摄影师的创意利器与调色宝典

在数码摄影的世界里&#xff0c;后期处理是摄影师们展现创意、调整细节、提升作品质量的重要步骤。而Nik Collection by DxO作为一款由DxO公司开发的强大照片编辑插件套件&#xff0c;为摄影师们提供了一套全面的、功能丰富的工具集&#xff0c;让他们的创意得以充分发挥。 Ni…

数据结构 - C/C++ - 栈

目录 结构特性 结构实现 结构容器 结构设计 顺序栈 链式栈 结构特性 栈(stack)是线性表的一种形式&#xff0c;限定仅在表的一端进行插入或者删除的操作。 栈顶 - 表中允许插入、删除的一端称为栈顶(top)&#xff0c;栈顶位置是可以发生变化的。 插入 - 进栈、入栈、压栈…

10月开始,所有新来日本的外国人都必须加入公共年金体系!

为了吸引更多外国人来日本工作并为他们提供更好的养老保障&#xff0c;日本厚生劳动省最近宣布了一项新政策。 从今年10月开始&#xff0c;所有新来日本的外国人都必须加入公共年金体系。 虽然之前已经有这个要求&#xff0c;但还是有不少人没加入。 因此&#xff0c;日本年金机…