智能指针: share_ptr(共享智能指针)

news2025/1/29 14:11:27

智能指针

c++中不像java自带垃圾回收机制,必须释放掉分配的内存,否则机会造成内存泄漏。因此c++11加入了智能指针。智能指针是存储指向动态分配(堆)对象指针的类,用于生存期的控制,能够确保在离开指针所在作用域时,自动的销毁分配的对象,防止内存泄漏。智能指针的核心实现技术是引用技术,每使用它一次内部引用计数+1,每析构一次内部引用技术-1,减为0时,删除原始指针指向的堆区内存,使用智能指针需要引用头文件<memory>.

 std::shared_ptr:共享的智能指针

 std::unique_ptr:独占的智能指针

 std::weak_ptr:弱引用的智能指针,它不共享指针,不能操作资源,是用来监视shared_ptr的,解决循环引用问题。

 share_ptr(共享智能指针)

 初始化:

 共智能指针是指多个智能指针可以同时管理同一块有效的内存,共享智能指针share_ptr是一个模板类,如果进行初始化有三种方式如下:

  通过构造函数初始化

 std::make_shared辅助函数

reset方法

 共享智能指针对象初始化完毕之后就指向了要管理的那块堆区内存,如果想要查看当前有多少个智能指针同时管理着这块内存可以使用共享指针提供的一个成员函数use_count.

构造函数初始化

构造函数初始化

//构造函数
shared_ptr<int> ptr1(new int(520));
cout<<"ptr1 管理的内存引用技术:"<<ptr1.use_count()<<endl;

//拷贝构造函数
shared_ptr<int> ptr2(ptr1);
cout<<"ptr2 管理的内存引用技术:"<<ptr2.use_count()<<endl;
shared_ptr<int>ptr3 = ptr1;
cout<<"ptr3管理的内存引用技术:"<<ptr3.use_count()<<endl;

//移动构造函数
shared_ptr<int> ptr4(std::move(ptr1));
cout<<"ptr4管理的内存引用技术:"<<ptr4.use_count()<<endl;
std::shared_ptr<int> ptr5 = std::move(ptr2);
cout<<"ptr5管理的内存引用技术:"<<ptr5.use_count()<<endl;

打印结果为:1    1   0    0

如果智能指针被初始化了一块有效内存,那么这块内存的引用技术+1,如果智能指针没有被初始化或者被初始化为nullptr空指针,引用计数为0,另外,不要使用一个原始指针初始化多个shared_ptr。

拷贝和移动构造函数初始化

当一个智能指针被初始化后,就可以通过这个智能指针初始化其他新的对象。在创建新对象的时候,对应的拷贝构造函数或者移动构造函数就被调用了

//构造函数
shared_ptr<int> ptr1(new int(520));
cout<<"ptr1管理的内存引用计数:"<<ptr1.use_count()<<endl;

//拷贝构造函数
shared_ptr<int> ptr2(ptr1);
cout<<"ptr2管理的内存引用计数:"<<ptr2.use_cont()<<endl;
shared_ptr<int> ptr3 = ptr1;
cout<<"ptr3管理的内存引用计数:"<<ptr3.use_count()<<endl;

//移动构造函数
shared_ptr<int> ptr4(std::move(ptr1));
cout<<"pptr4管理的内存引用计数:"<<ptr4.use_count()<<endl;
std::shared_ptr<int> ptr5 = std::move(ptr2);
cout<<"ptr5管理的内存引用计数:"<<ptr5.use_count()<<endl;

打印结果: 1   2     3      3     3

如果使用拷贝的方式初始化共享智能指针,这两个对象会同时管理同一块内存,堆内存对应的引用技术也会增加。如果使用移动构造的方式初始化智能指针对象,只是转让了内存的所有权,管理内存的对象不会增加,因此内存引用计数不会增加。

 std::make_shared 初始化

通过c++11提供的std::make_shared()就可以完成内存对象的创建并将其初始化给智能指针

#include<iostream>
#include<string>
#include<memory>
using namespace std;

class Test
{
public:
    Test()
    {
        cout<<"无参构造函数"<<endl;
    }
    Test(int x)
    {
        cout<<"int 类型构造函数"<<x<<endl;
    }
    Test(string str)
    {
        cout<<"string类型的构造函数"<<str<<endl;
    }
    ~Test()
    {
        cout<<"析构函数"<<endl;
    }
};

int main()
{
    //使用智能指针管理一快int型的堆内存,内部引用计数为1
    shared_ptr<int> ptr1 = make_shared<int>(520);
    cout<<"ptr1管理的内存引用计数:"<<ptr.use_count()<<endl;

    shared_ptr<Test> ptr2 = make_shared<Test>();
    cout<<"ptr2管理的内存引用计数:"<<ptr2.use_cont()<<endl;

    shared_ptr<Test> ptr3 = make_shared<Test>(520);
    cout<<"ptr3管理的内存引用计数:"<<ptr3.use_count()<<endl;

    shared_ptr<Test> ptr3 = make_shared<Test>(QQQ);
    cout<<"ptr4管理的内存引用计数:"<<ptr4.use_count()<<endl;
}

 

#include<iostream>
#include<string>
#include<memory>
using namespace std;

int main()
{
    //使用智能指针管理一块int型的堆内存,内部引用计数为1
    shared_ptr<int> ptr1 = make_shared<int>(520);
    shared_ptr<int> ptr2 = ptr1;
    shared_ptr<int> ptr3 = ptr1;
    shared_ptr<int> ptr4 = ptr1;
    cout<<"ptr1管理的内存引用计数:"<<ptr1.use_count()<<endl;
    cout<<"ptr2管理的内存引用计数:"<<ptr2.use_count()<<endl;
    cout<<"ptr3管理的内存引用计数:"<<ptr3.use_count()<<endl;
    cout<<"ptr4管理的内存引用计数:"<<ptr4.use_count()<<endl;

    ptr4.reset();
    cout<<"ptr1管理的内存引用计数:"<<ptr1.use_count()<<endl;
    cout<<"ptr2管理的内存引用计数:"<<ptr2.use_count()<<endl;
    cout<<"ptr3管理的内存引用计数:"<<ptr3.use_count()<<endl;
    cout<<"ptr4管理的内存引用计数:"<<ptr4.use_count()<<endl;

    shared_ptr<int> ptr5;
    ptr5.reset(new int(250));
    cout<<"ptr5管理的内存引用计数:"<<ptr5.use_count()<<endl;
    return 0;
}

 

对于一个未初始化的共享智能指针,可以通过reset方法来初始化,当智能指针中有值的时候,调用reset会使引用计数-1.

获取原始指针

 get()函数

int main()
{
    shared_ptr<int> p(new int);
    *p = 100;
    cout<<*p.get()<<""<<*p<<endl;
    return 0;
}
100 100

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

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

相关文章

Selenium4自动化框架(超级详细)

目录 Selenium4 安装Selenium 安装浏览器驱动 实战案例 导入模块及浏览器驱动 导入模块 启动驱动 定位元素 id、name、class定位 tag_name定位 xpath定位 css选择器定位 link_text、partial_link_text定位 其他定位 定位一组元素 执行操作 浏览器操作 获取信息…

1. 自然语言处理NLP-数据预处理

NLP任务预处理的流程包括&#xff1a; 收集语料库、文本清洗、分词、去掉停用词、标准化和特征提取等。 &#xff08;1&#xff09;收集语料库 &#xff08;2&#xff09;清洗数据 eg&#xff1a;删除所有不相关的字符&#xff0c;例如非字母数字字母 &#xff08;3&#xff09…

Java Servlet相关面试题

一、什么是servlet&#xff1f; Servlet是运行在java服务器中的小型Java程序。 作用&#xff1a;接收用户请求&#xff0c;并对请求作出处理&#xff0c;将处理结果相应给客户端。 Servlet是JavaWeb三大组件&#xff08;Servlet、过滤器&#xff0c;监听器 &#xff09;之一…

C++学习day--12 循环

第 1 节: 需求分析、项目实现——重复验证 项目实现&#xff1a; #include <iostream> #include <Windows.h> #include <string> using namespace std; int main(void) { string name; string pwd; while (1) { system("cls"); std::cout <…

Windows编辑开发中的内聚性、内聚类型、耦合性和耦合类型

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天总结一下Windows编辑开发中的内聚性、内聚类型、耦合性和耦合类型。 软件设计的基本原则是信息隐蔽性与模块独立性。 模块设计目标是高内聚&#xff0c;低耦合。 然后记住下面这张神图&#xff0c;一张…

Mybatis之MetaObject

在mybatis中&#xff0c;ResultSetHandler在收集JDBC返回的结果后需要转换成对应的Bean对象&#xff0c;其实映射的原理基本大家都能想到使用的时候java中的反射机制&#xff0c;但是在Mybatis中&#xff0c;提供了一个更加强大的对象&#xff0c;就是MetaObject&#xff0c;使…

Python - 面向对象编程 - 实例方法、静态方法、类方法

实例方法 在类中定义的方法默认都是实例方法&#xff0c;前面几篇文章已经大量使用到实例方法 实例方法栗子 class PoloBlog:def __init__(self, name, age):print("自动调用构造方法")self.name nameself.age agedef test(self):print("一个实例方法&…

阿里P8写出的《深入理解Java虚拟机》最新版,轻松学会JVM底层

前言 Java是目前用户最多、使用范围最广的软件开发技术&#xff0c;Java的技术体系主要由支撑Java程序运行的虚拟机、提供各开发领域接口支持的Java类库、Java编程语言及许许多多的第三E方Java框架(如Spring、 MyBatis等) 构成。在国内&#xff0c;有关Java类库API、Java语言语…

能源汽车下乡充电桩建设优化建议及解决方案

安科瑞 耿敏花 摘 要&#xff1a;5月5日&#xff0c;国务院总理李强主持召开国务院常务会议&#xff0c;审议通过关于加快发展先进制造业集群的意见&#xff0c;部署加快建设充电基础设施&#xff0c;更好支持新能源汽车下乡和乡村振兴。 会议指出&#xff0c;农村新能源汽车市…

JavaScript 原型和原型链

文章目录 JavaScript 原型和原型链概述new操作符的流程原型对象、构造函数、实例关系原型链原型链查找hasOwnPropertyin JavaScript 原型和原型链 概述 任何函数都有prototype属性&#xff0c;prototype是原型的意思。 prototype的属性值是个对象&#xff0c;它默认拥有cons…

Java内存模型(Java Memory Mode,JMM)

并发编程模型的两个关键问题 线程之间如何通信及线程之间如何同步。 线程之间如何通信&#xff1a;共享内存&#xff0c;消息传递线程之间如何同步通信是指线程之间以何种机制来 交换信息同步是指程序中用于控制不同线程间 操作发生相对顺序 的机制在共享内存的并发模型里&a…

网络安全大厂面试题汇总

注&#xff1a;本套面试题&#xff0c;已整理成pdf文档&#xff0c;但内容还在持续更新中&#xff0c;因为无论如何都不可能覆盖所有的面试问题&#xff0c;更多的还是希望由点达面&#xff0c;查漏补缺。 ​ 某大厂一面整理 1、防范常见的 Web 攻击 2、重要协议分布层 3、ar…

c++使用yaml -基于windows10

参考&#xff1a;Windows10下使用VS2017编译和使用yaml-cpp库_雪域迷影的博客-CSDN博客 1. 下载yaml-cpp 建议在github下载其最新的官方版本&#xff0c;不要在其他平台下载该工具软件&#xff0c;下载地址如下&#xff08;其中的一个版本&#xff09;&#xff1a; Release …

理解深度可分离卷积

1、常规卷积 常规卷积中&#xff0c;连接的上一层一般具有多个通道&#xff08;这里假设为n个通道&#xff09;&#xff0c;因此在做卷积时&#xff0c;一个滤波器&#xff08;filter&#xff09;必须具有n个卷积核&#xff08;kernel&#xff09;来与之对应。一个滤波器完成一…

Android自定义键盘(KeyboardView)

目录 1.场景&#xff1a;2.想法&#xff1a;3.开始实现&#xff1a;《一》 在res包下创建xml目录,Keyboard标签来定义键盘布局&#xff1a;《二》创建IKeyboardView类并继承KeyboardView类,设置键盘布局&#xff08;数字和字母&#xff09;《三》 处理自定义键盘按键的点击事件…

C++11:列表初始化、新增关键字和新增的默认成员函数

目录 一. 列表初始化 1.1 {}列表初始化的方法 1.2 列表初始化实现的原理 二. C11新增关键字 2.1 decltype -- 自动类型推断 2.2 nullptr -- 空指针 2.3 default -- 强制生成默认成员函数 2.4 delete -- 禁止生成默认成员函数 2.5 final -- 禁止类被继承/禁止虚函数被重…

自动化测试之PageObject设计模式

译文如下&#xff1a; PageObject 当您针对网页编写测试时&#xff0c;您需要参考该网页中的元素以单击链接并确定显示的内容。但是&#xff0c;如果您编写直接操作 HTML 元素的测试用例&#xff0c;则您的测试将无法应对 UI 中的频繁修改。PageObject对应于一个HTML网页、页…

5月26号软件资讯更新合集......

Windows Terminal 1.18 新功能预览&#xff1a;标签拖拽、上下文菜单... Windows Terminal 预览版已更新到 1.18 版本&#xff0c;带来多项实用内容&#xff0c;一起来看看这个版本的新东西&#xff1a; 标签撕裂&#xff08;拖拽功能&#xff09; Windows Terminal 已支持对…

安装Redis6

安装Redis 安装redis依赖 redis是基于C语言编写的&#xff0c;因此首选需要安装redis所需的gcc依赖 yum install -y gcc tcl 上传安装包并解压 我安装的是redis-6.2.6版本&#xff0c;并且放到了/usr/local/src目录下 - 进入/usr/local/src目录shellcd /usr/local/src解压 tar …

全球最受欢迎低代码平台排行榜出炉

低代码平台正在成为寻求快速有效地构建应用程序的企业的首选解决方案。这些平台减少了编码要求&#xff0c;使企业能够在降低成本的同时更快地完成应用程序开发项目。在本文中&#xff0c;将探索全球受欢迎的低代码平台排行榜。 该排名使用的标准包括易用性、成本效益、集成性、…