C++ —— 智能指针 unique_ptr (上)

ops/2025/1/31 6:39:24/

C++ —— 智能指针 unique_ptr (上)

  • 普通指针的不足
  • 普通指针的释放
  • 智能指针
  • 智能指针 unique_ptr
    • 智能指针初始化
    • 错误用法
    • get()方法返回裸指针
    • 智能指针不支持指针的运算(+、-、++、- -)

普通指针的不足

  • newnew [] 的内存需要用deletedelete []释放(堆区的内存一定要手工释放,否则会发生内存的泄露);
  • 程序员主观上的失误,忘记或漏掉释放;
  • 不确定何时释放。

普通指针的释放

  • 类内指针,在析构函数释放
  • 堆区的内存是C++内置数据类型,没有析构函数,只能delete释放;
  • new出来的类,还是得用delete释放。

智能指针

智能指针的目的:解决资源释放的问题。

智能指针使用步骤:

  • 智能指针是类模板,在栈上创建智能指针对象;
  • 把普通指针交给智能指针对象;
  • 智能指针对象过期时,调用析构函数释放普通指针的内存。

C++11标准的智能指针类型:unique_ptrshared_ptrweak_ptr

智能指针 unique_ptr

C++中,多个指针可以指向同一个对象。
unique_ptr 独享它指向的对象,也就是说,同时unique_ptr指向同一个对象,当这个unique_ptr销毁时,指向对象也随即被销毁

示例代码如下:

#include <iostream>
#include <memory> // 使用智能指针需要包含的头文件
using namespace std;class A {
public:string m_name;A() {cout << "A()" << endl;}A(const string& name): m_name(name) {cout << "A(const string&)" << endl;}~A() {cout << "~A()" << endl;}
};int main () {A* pa = new A("aaa");// delete pa;return 0;
}

因为没有delete,所以只有构造函数的日志信息,运行结果如下:

A(const string&)

使用智能指针unique_ptr来管理普通指针pa代码如下:

int main () {A* pa = new A("aaa");unique_ptr<A> pu_a(pa); // 间接地让智能指针 pu_a 来管理对象// 需要管理的普通指针的基类型是 A(也就是模板参数)// pa 是被管理的指针,pa 指向了 new 出来的对象的地址。return 0;
}

运行效果如下:

A(const string&)
~A()

可以看到,尽管没有使用delete语句,也销毁了A的对象。原因是:智能指针,它有析构函数,在它的析构函数中,使用了delete语句。

可以像使用普通指针一样去使用智能指针,代码如下:

int main () {A* pa = new A("aaa");// delete pa;unique_ptr<A> pu_a(pa);cout <<"m_name = " << (*pa).m_name << endl;cout << "m_name = " << pa->m_name << endl;cout << "m_name = " << (*pu_a).m_name << endl;cout << "m_name = " << pu_a->m_name << endl;return 0;
}

运行效果如下:

A(const string&)
m_name = aaa
m_name = aaa
m_name = aaa
m_name = aaa
~A()

智能指针初始化

  • 方法一(常用):
    unique_ptr<A> pu_a(new A("abcd")); // 分配内存并初始化
int main () {unique_ptr<A> pu_a(new A("abcd"));// new 返回的是对象的地址cout <<"m_name = " << (*pu_a).m_name << endl;cout << "m_name = " << pu_a->m_name << endl;return 0;
}

运行效果如下:

A(const string&)
m_name = abcd
m_name = abcd
~A()

  • 方法二:
    unique_ptr<A> p = make_unique<A>("abcd"); // C++14标准
    unique_ptr<int> p1=make_unique<int>(); // 数据类型为int
    unique_ptr<A> p2 = make_unique<A>(); // 数据类型为A,默认构造函数
    unique_ptr<A> p3 = make_unique<A>("abcd"); // 数据类型为A,一个参数的构造函数
    unique_ptr<A> p4 = make_unique<A>("abcd","efgh"); // 数据类型为A,两个参数的构造函数
  • 方法三(不推荐):
    A* p = new A("abcd");
    unique_ptr<A> pu (p); // 用已存在的地址初始化

错误用法

A* p = new a("abcd");
unique_ptr<A> pu1 = p; // 错误,不能把普通指针直接赋给智能指针。
unique_ptr<A> pu2 = new A("abcd"); // 错误,不能把普通指针直接赋给智能指针。
unique_ptr<A> pu3 = pu2; // 错误,不能用其它unique_ptr拷贝构造。
unique_ptr<A> pu4;
pu4 = pu1; // 错误,不能用 = 对 unique_ptr 进行赋值。
// 裸指针就是普通指针 pa就是普通指针,也叫裸指针
A* pa = new A("efgh");
unique_ptr<A> pu_a1(pa);
unique_ptr<A> pu_a2(pa);
unique_ptr<A> pu_a3(pa);
// 程序会异常退出,原因是:多个unique_ptr对象对同一块内存释放了多次

get()方法返回裸指针

int main () {A* pa = new A("efgh");unique_ptr<A> pu_a(pa);cout << "裸指针的值是:" << pa << endl;cout << "pu_a.get() = " << pu_a.get() << endl;// cout << pa->m_name << endl; // efgh// cout << pu_a.get()->m_name << endl; // efghcout << "pu_a的地址:" << &pu_a << endl;// pu_a是unique_ptr<A>模板类创建的对象,有自己的地址。// 自己的地址和它管理的原始指针的地址不是一回事return 0;
}

运行结果如下:

A(const string&)
裸指针的值是:0x557320cf6eb0
pu_a.get() = 0x557320cf6eb0
pu_a的地址:0x7ffde063e680
~A()

智能指针不支持指针的运算(+、-、++、- -)

感谢浏览,一起学习


http://www.ppmy.cn/ops/154409.html

相关文章

搜索引擎快速收录:关键词布局的艺术

本文来自&#xff1a;百万收录网 原文链接&#xff1a;https://www.baiwanshoulu.com/21.html 搜索引擎快速收录中的关键词布局&#xff0c;是一项既精细又富有策略性的工作。以下是对关键词布局艺术的详细阐述&#xff1a; 一、关键词布局的重要性 关键词布局影响着后期页面…

redis的分片集群模式

redis的分片集群模式 1 主从哨兵集群的问题和分片集群特点 主从哨兵集群可应对高并发写和高可用性&#xff0c;但是还有2个问题没有解决&#xff1a; &#xff08;1&#xff09;海量数据存储 &#xff08;2&#xff09;高并发写的问题 使用分片集群可解决&#xff0c;分片集群…

vim操作简要记录

操作容易忘记&#xff0c;记录一下基本使用的 :wq保存退出 :w :q :q! :wq! i I a A 方向键 h左 j下 k上 l右 dd删除方行&#xff08;这其实是剪切行操作&#xff0c;不过一般用作删除&#xff0c;长按可删除&#xff0c;不过按.执行上一次操作删除更快&#xff09; .执行上…

LeetCode 0040.组合总和 II:回溯 + 剪枝

【LetMeFly】40.组合总和 II&#xff1a;回溯 剪枝 力扣题目链接&#xff1a;https://leetcode.cn/problems/combination-sum-ii/ 给定一个候选人编号的集合 candidates 和一个目标数 target &#xff0c;找出 candidates 中所有可以使数字和为 target 的组合。 candidates…

使用Ollama和Open WebUI快速玩转大模型:简单快捷的尝试各种llm大模型,比如DeepSeek r1

Ollama本身就是非常优秀的大模型管理和推理组件&#xff0c;再使用Open WebUI更加如虎添翼&#xff01; Ollama快速使用指南 安装Ollama Windows下安装 下载Windows版Ollama软件&#xff1a;Release v0.5.7 ollama/ollama GitHub 下载ollama-windows-amd64.zip这个文件即可…

OpenAI-Edge-TTS:本地化 OpenAI 兼容的文本转语音 API,免费高效!

文本转语音&#xff08;TTS&#xff09;技术已经成为人工智能领域的重要一环&#xff0c;无论是语音助手、教育内容生成&#xff0c;还是音频文章创作&#xff0c;TTS 工具都能显著提高效率。今天要为大家介绍的是 OpenAI-Edge-TTS&#xff0c;一款基于 Microsoft Edge 在线文本…

让Android adb支持互联网调试脱离局域网

某些特殊场景下由于不方便&#xff0c;手机不在身边&#xff0c;但需要进行adb调试。 首先可以先开启adb的无线调试模式&#xff0c;我使用的是第二种方式。 在Android手机上安装一个终端模拟器&#xff0c;并赋予root权限&#xff0c;随后执行&#xff1a; setprop service.…

React 前端开发解析:从核心概念到最佳实践

引言 React 作为当今最流行的前端框架之一&#xff0c;凭借其组件化、声明式编程和高效的虚拟 DOM 机制&#xff0c;彻底改变了现代 Web 开发的范式。无论是构建小型应用还是复杂的企业级系统&#xff0c;React 都展现出了强大的灵活性和可扩展性。本文将深入探讨 React 的核心…