[c++] 智能指针(shared_ptr, weak_ptr, unique_ptr)

c++ 中的智能指针让 c++ 看起来像 java,go 这种带 GC 的语言,但和 GC 又不完全相同。c++ 中的智能指针使用引用计数,当引用计数减为 0 的时候就会立即释放资源,释放资源具有实时性;而像 java,go 这样的 GC 语言,当对象不被引用时,可能不会立即释放资源,资源的释放有滞后性。 

如果不使用智能指针,需要开发者使用 new 和 delete 创建和销毁对象。

使用智能指针,开发者不需要手动来释放资源,当引用计数变为 0 时自动释放。

就这一个好处。

使用智能指针需要包含头文件 memory。

1 shared_ptr 和 weak_ptr

1.1 shared_ptr

shared_ptr 是共享指针,对于一个对象来说,可以有多个 shared_ptr 指针指向它,每多一个指针指向它,这个对象的引用计数就会增加 1。shared_ptr 有一个函数是 reset(),每 reset() 一次,对象的引用计数就会减去 1,当引用计数变为 0 的时候,就会释放对象。

shared_ptr 的常用方法:

use_count()获取对象的引用计数
get()获取对象的原始指针
reset()引用计数减 1,或者使用一个新的对象重置这个智能指针

shared_ptr 示例代码如下:

#include <iostream>
#include <string>
#include <memory>class Test {
public:Test() {std::cout << "Test()" << std::endl;}~Test() {std::cout << "~Test()" << std::endl;}void Do() {std::cout << "Test() Do()" << std::endl;}
};void SharedParam(std::shared_ptr<Test> t) {std::cout << "SharedParam, t.use_count() = " << t.use_count() << std::endl;
}void SharedRefParam(std::shared_ptr<Test> &t) {std::cout << "SharedRefParam, t.use_count() = " << t.use_count() << std::endl;
}int main() {std::shared_ptr<Test> t1 = std::make_shared<Test>();std::shared_ptr<Test> t2 = t1;std::shared_ptr<Test> t3 = t1;std::cout << "t1.use_count() = " << t1.use_count() << std::endl;std::cout << "t2.use_count() = " << t2.use_count() << std::endl;std::cout << "t3.use_count() = " << t3.use_count() << std::endl;std::cout << std::endl;SharedParam(t1);SharedRefParam(t1);t3.reset();std::cout << "after t3 reset" << std::endl;std::cout << "t1.use_count() = " << t1.use_count() << std::endl;std::cout << "t2.use_count() = " << t2.use_count() << std::endl;std::cout << "t3.use_count() = " << t3.use_count() << std::endl;std::cout << std::endl;t2.reset();t1.reset();std::cout << "after t2, t1 reset" << std::endl;std::cout << "t1.use_count() = " << t1.use_count() << std::endl;std::cout << "t2.use_count() = " << t2.use_count() << std::endl;std::cout << "t3.use_count() = " << t3.use_count() << std::endl;std::cout << std::endl;std::shared_ptr<Test> t4;t4.reset(new Test);std::cout << "t2.use_count() = " << t2.use_count() << std::endl;std::cout << "t4.use_count() = " << t4.use_count() << std::endl;std::cout << std::endl;Test *t5 = t4.get();t5->Do();return 0;
}

 代码编译之后,运行结果如下:

1.2 weak_ptr

1.2.1 shared_ptr 存在的问题:循环引用,无法自动释放资源

要了解为什么需要 weak_ptr 就需要了解 shared_ptr 存在的问题。

shared_ptr 的问题示例代码如下,按我们的理解,main() 函数返回的时候,a,b 两个共享指针会自动释放。代码中即使分别对 a 和 b 进行了 reset,也不能自动释放。

#include <iostream>
#include <string>
#include <memory>class B;class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}std::shared_ptr<B> b_;
};class B {
public:B() {std::cout << "B()" << std::endl;}~B() {std::cout << "~B()" << std::endl;}std::shared_ptr<A> a_;
};int main() {std::shared_ptr<A> a = std::make_shared<A>();std::shared_ptr<B> b = std::make_shared<B>();a->b_ = b;b->a_ = a;a.reset();b.reset();return 0;
}

运行结果如下,类 A 和类 B 的析构函数并没有被调用。

main() 函数返回时的析构过程:

(1)a 和 b 析构,a 和 b 的引用计数减 1

(2)销毁 a 中的 b,这个时候 b 的引用计数要减为 0,减为 0 的时候就要销毁 b 中的成员 a;销毁 b 中的 a,同理 b 中的 a 的引用计数要减为 0。两者死锁,最终到是 a 和 b 都无法析构。

1.2.2 weak_ptr

weak_ptr 不会增加对象的引用计数,当有循环引用这种情况时,可以使用 weak_ptr 来代替 shared_ptr,以此来解决 shared_ptr 不能自动释放的问题。

本来 shared_ptr 是用来简化开发工作的,这里又引入了 weak_ptr,通过 weak_ptr 又不能直接访问对象,还要使用 lock 来获取对象。开发工作并没有简化多少。

本来 c++ 中的引用是为了减少指针操作的,但是 c++ 中的引用又分左值引用,右值引用,导致用起来并不比指针简单。

代码如下所示,A 中对 B 的引用由  shared_ptr 改成了 weak_ptr。

#include <iostream>
#include <string>
#include <memory>class B;class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}std::weak_ptr<B> b_;
};class B {
public:B() {std::cout << "B()" << std::endl;}~B() {std::cout << "~B()" << std::endl;}std::shared_ptr<A> a_;
};int main() {std::shared_ptr<A> a = std::make_shared<A>();std::shared_ptr<B> b = std::make_shared<B>();a->b_ = b;b->a_ = a;int a_use_count = a.use_count();int b_use_count = b.use_count();std::cout << "a use count = " << a_use_count << ", b use count = " << b_use_count << std::endl;a.reset();b.reset();return 0;
}

代码编译之后,运行结果如下,可以看到 A 和 B 的析构函数都被调用了。

因为 A 中对 B 的引用是 weak_ptr,weak_ptr 不会增加引用计数,所以 b 的引用计数是 1。

通过 weak_ptr 不能直接访问对象,要通过 lock() 获取 shared_ptr,lock() 不为空,则可用;为空,说明对象已经释放了。通过 expired() 方法也能确定 shared_ptr 是不是已经过期了。

#include <iostream>
#include <string>
#include <memory>class B;class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};int main() {std::shared_ptr<A> a = std::make_shared<A>();std::weak_ptr<A> b = a;std::cout << "1, a use count = " << a.use_count() << ", b use count = " << b.use_count() << std::endl;a.reset();std::cout << "2, a use count = " << a.use_count() << ", b use count = " << b.use_count() << std::endl;if (b.lock()) {std::cout << "shared a exists" << std::endl;} else {std::cout << "shared a not exists" << std::endl;}std::cout << "b.expired() = " << b.expired() << std::endl;a.reset(new A);b = a;std::cout << "3, a use count = " << a.use_count() << ", b use count = " << b.use_count() << std::endl;b.reset();std::cout << "4, a use count = " << a.use_count() << ", b use count = " << b.use_count() << std::endl;if (b.lock()) {std::cout << "shared a exists" << std::endl;} else {std::cout << "shared a not exists" << std::endl;}b = a;if (b.lock()) {auto sp = b.lock();sp->Do();}return 0;
}

运行结果如下:

 

 

2 unique_ptr

unique_ptr 是独占智能指针,永远只能有一个指针指向这个对象。

在不同智能指针间传递需要通过 std::move() 来进行。

示例代码如下所示:

#include <iostream>
#include <string>
#include <memory>class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};int main() {std::unique_ptr<A> u1 = std::make_unique<A>();std::unique_ptr<A> u2(new A);std::unique_ptr<A> u3(std::make_unique<A>());if (u1) {std::cout << "u1 Do()" << std::endl;u1->Do();}u1.reset();if (u1) {std::cout << "u1 Do() after u1.reset()" << std::endl;}if (u2) {std::cout << "u2 Do()" << std::endl;u2->Do();}std::unique_ptr<A> u4 = std::move(u2);if (u2) {std::cout << "u2 Do()i after u2 moved" << std::endl;u2->Do();}return 0;
}

 

unique_ptr 自己调用 reset() 或者使用 std::move() 转移之后,自己就会变为 nullptr。

 

3 自动析构

(1)[普通对象] 全局和局部的对象,在函数返回时析构

#include <iostream>
#include <string>
#include <memory>class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};A a1;
void Test() {A a2;
}int main() {std::cout << "main" << std::endl;Test();A a3;return 0;
}

(2)[普通对象] 对象指针不会自动释放,需要调用 delete 手动释放

#include <iostream>
#include <string>
#include <memory>class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};A *a1;
void Test() {A *a2 = new A;
}int main() {std::cout << "main" << std::endl;Test();A *a3 = new A;return 0;
}

(3)[普通对象] 对象释放的时候,对象中的对象成员会自动释放

#include <iostream>
#include <string>
#include <memory>class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};class B {
public:B() {std::cout << "B()" << std::endl;}~B() {std::cout << "~B()" << std::endl;}A a;
};int main() {std::cout << "main" << std::endl;A a1;return 0;
}

(4)[智能指针] 全局智能指针,局部智能指针,对象中的智能指针会自动释放

#include <iostream>
#include <string>
#include <memory>class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};class B {
public:B() {std::cout << "B()" << std::endl;}~B() {std::cout << "~B()" << std::endl;}std::shared_ptr<A> a1;std::unique_ptr<A> a2;
};std::shared_ptr<A> sa1;
std::unique_ptr<A> ua1;int main() {std::cout << "main" << std::endl;sa1 = std::make_shared<A>();ua1 = std::make_unique<A>();std::shared_ptr<A> sa2 = std::make_shared<A>();std::unique_ptr<A> ua2 = std::make_unique<A>();B b;b.a1 = std::make_shared<A>();b.a2 = std::make_unique<A>();return 0;
}

(5)[智能指针] 如果智能指针所在的对象或者容器不会自动析构,那么智能指针也不能析构

#include <iostream>
#include <string>
#include <memory>
#include <vector>class A {
public:A() {std::cout << "A()" << std::endl;}~A() {std::cout << "~A()" << std::endl;}void Do() {std::cout << "Do()" << std::endl;}
};class B {
public:B() {std::cout << "B()" << std::endl;}~B() {std::cout << "~B()" << std::endl;}std::shared_ptr<A> a;
};int main() {B *b = new B;b->a = std::make_shared<A>();return 0;
}

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

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

相关文章

软考中项- 3.5软件架构;3.6典型应用集成技术

文章目录 1. C/S模式&#xff1a;基于资源不对等&#xff0c;为实现共享而提出的模式。2. 数据库和数据仓库的区别3. 关于软件分层架构描述4. 管道/过滤器模式5. javaEE的运行环境6、 针对目前广泛使用的分布式应用&#xff0c;其软件架构设计需要考虑如下问题&#xff1a;7. w…

【蓝桥杯】拓扑排序

一.拓扑排序 1.定义&#xff1a; 设G&#xff08;V&#xff0c;E&#xff09;是一个具有n个顶点的有向图&#xff0c;V中的顶点序列称为一个拓扑序列&#xff0c;当且仅当满足下列条件&#xff1a;若从顶点到有一条路径&#xff0c;则在顶点序列中顶点必在之前。 2.基本思想…

DVWA 靶场之 Command Injection(命令执行)原理介绍、分隔符测试、后门写入与源码分析、修复建议

在打靶之前我们需要先解决一个乱码问题 参考我之前的博客&#xff1a; 关于DVWA靶场Command Injection&#xff08;命令注入&#xff09;乱码的解决方案-CSDN博客 简单介绍一下命令执行漏洞&#xff1a; 命令执行漏洞是一种常见的网络安全漏洞&#xff0c;它允许攻击者通过向应…

【计算机网络】应用层自定义协议

自定义协议 一、为什么需要自定义协议&#xff1f;二、网络版计算器1. 基本要求2. 序列化和反序列化3. 代码实现&#xff08;1&#xff09;封装 socket&#xff08;2&#xff09;定制协议和序列化反序列化&#xff08;3&#xff09;客户端&#xff08;4&#xff09;计算器服务端…

SAP PO接口行项目json缺少中括号[]问题

PO接口小问题问题&#xff1a;如果需要同时传输DATA与ITEM&#xff0c;此处选择很重要&#xff0c;如果选择&#xff1a;HTTP Header ITEM将缺少[].需要注意 PO接口小问题 问题&#xff1a;如果需要同时传输DATA与ITEM&#xff0c;此处选择很重要&#xff0c;如果选择&#…

浏览器跨 Tab 窗口通信原理

前言 原文地址&#xff1a;浏览器跨 Tab 窗口通信原理及应用实践 作者是Chokcoco 一位css大牛。之前就从大佬的文章中学到了不少东西&#xff0c;最近大佬写了篇 浏览器跨 Tab 窗口通信原理及应用实践 感觉挺有意思的&#xff0c;自己打算学习记录一下。 文章中提出了三种方…

leetcode hot100 买卖股票的最佳时机二

注意&#xff0c;本题是针对股票可以进行多次交易&#xff0c;但是下次买入的时候必须保证上次买入的已经卖出才可以。 动态规划可以解决整个股票买卖系列问题。 dp数组含义&#xff1a; dp[i][0]表示第i天不持有股票的最大现金 dp[i][1]表示第i天持有股票的最大现金 递归公…

RestTemplate启动问题解决

⭐ 作者简介&#xff1a;码上言 ⭐ 代表教程&#xff1a;Spring Boot vue-element 开发个人博客项目实战教程 ⭐专栏内容&#xff1a;个人博客系统 ⭐我的文档网站&#xff1a;http://xyhwh-nav.cn/ RestTemplate启动问题解决 问题&#xff1a;在SpringCloud架构项目中配…

C++之多态

目录 一&#xff0c;概念等基础 1&#xff09;多态的定义 2&#xff09;虚函数 3&#xff09;构成多态的条件 4&#xff09;虚函数的作用 二&#xff0c;多态原理 1&#xff09;虚指针 2&#xff09;如何拿到这个虚指针 3&#xff09;到底如何实现多态 一&#xff0c;概…

Vue ElementUI 修改消息提示框样式—messageBox 的大小

在窄屏模式下&#xff08;移动端或pda&#xff09;&#xff0c;提示框的宽度太宽&#xff0c;会出现显示不完全的问题。 应当如何修改 ElementUI 的样式呢&#xff1f; open() {this.$confirm(window.vm.$i18n.t("tips.conLogOut"),window.vm.$i18n.t("tips.tip…

使用百度地图api根据输入的过个经纬度进行轨迹绘制并且可以标记

使用百度地图api根据输入的过个经纬度进行轨迹绘制并且可以标记 功能效果展示代码功能说明 功能效果展示 代码 <!DOCTYPE html> <html> <head><meta charset"utf-8"><title>根据经纬度绘制轨迹图</title><script type"…

K8S—集群调度

目录 前言 一 List-Watch 1.1 list-watch概述 1.2 list-watch工作机制 二 集群调度 2.1 调度过程 2.2 Predicate 和 Priorities 的常见算法和优先级选项 2.3 调度方式 三 亲和性 3.1 节点亲和性 3.2 Pod 亲和性 3.3 键值运算关系 3.4 Pod亲和性与反亲和性 3.5 示例…

golang学习1,dea的golang-1.22.0

参考&#xff1a;使用IDEA配置GO的开发环境备忘录-CSDN博客 1.下载All releases - The Go Programming Language (google.cn) 2.直接next 3.window环境变量配置 4.idea的go插件安装 5.新建go项目找不到jdk解决 https://blog.csdn.net/ouyang111222/article/details/1361657…

PCIE1—快速实现PCIE接口上下位机通信(一)

1.简介 PCI Express&#xff08;PCIE&#xff09;是一种高速串行总线标准&#xff0c;广泛应用于计算机系统中&#xff0c;用于连接主板和外部设备。在FPGA领域中&#xff0c;PCIE也被广泛应用于实现高速数据传输和通信。FPGA是一种灵活可编程的集成电路&#xff0c;可以根据需…

碳素光,碳光子,碳光灸 ,太阳灯 仪器

碳素光线疗法&#xff1a; 中西医、民间疗法融为一体&#xff0c;提高机体自身治愈力&#xff0c;免疫力&#xff0c;改善体质和保持健康&#xff0c;有助于疾病的预防和治疗的疗法。不吃药、不打针、不手术也能得健康&#xff0c;无任何副作用的自然物理疗法。 碳素光线仪市…

SCI一区 | Matlab实现ST-CNN-MATT基于S变换时频图和卷积网络融合多头自注意力机制的多特征分类预测

SCI一区 | Matlab实现ST-CNN-MATT基于S变换时频图和卷积网络融合多头自注意力机制的故障多特征分类预测 目录 SCI一区 | Matlab实现ST-CNN-MATT基于S变换时频图和卷积网络融合多头自注意力机制的故障多特征分类预测效果一览基本介绍模型描述程序设计参考资料 效果一览 基本介绍…

《Docker 简易速速上手小册》第8章 Docker 在企业中的应用(2024 最新版)

文章目录 8.1 Docker 在开发环境中的应用8.1.1 重点基础知识8.1.2 重点案例&#xff1a;Python Web 应用开发环境8.1.3 拓展案例 1&#xff1a;Python 数据分析环境8.1.4 拓展案例 2&#xff1a;Python 自动化测试环境 8.2 Docker 在生产环境的实践8.2.1 重点基础知识8.2.2 重点…

【嵌入式学习】IO进程线程day02.24

一、思维导图 二、习题 #define MSGSIZE sizeof(struct msgbuf)-sizeof(long) int main(int argc, const char *argv[]) {//创建子进程pid_t pidfork();//在父进程实现读功能if(pid>0){//1、创建key值key_t key 0;if((keyftok("/", k)) -1){perror("ftok …

【C++】STL容器之string(修改操作)

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

做了个很牛的网站,可以搜索网站的网站到底有多好用?

今天给大家推荐的网站叫做&#xff1a;毒蘑菇 - 搜索 毒蘑菇搜索&#xff0c;顾名思义呢&#xff0c;搜索的功能比较好用&#xff0c;大家上网的时候总是需要记住网站的地址&#xff0c;即使你知道网站的名称&#xff0c;也得跳转到百度然后在搜索&#xff0c;有时候百度上那么…