C++进阶语法——智能指针【学习笔记(五)】

文章目录

      • 1、智能指针简介
        • 1.1 原始指针(raw pointer)的⼀些问题
        • 1.2 智能指针(smart pointers)
      • 2、智能指针(smart pointers)——unique_ptr
        • 2.1 unique_ptr 的声明
        • 2.2 unique_ptr 的函数
        • 2.3 ⾃定义类型使⽤ unique_ptr
        • 2.4 unique_ptr 不⽀持拷⻉、赋值
        • 2.5 使⽤make_unique初始化(C++14标准)
        • 代码:
      • 3、智能指针(smart pointers)——shared_ptr
        • 3.1 shared_ptr的声明
        • 3.2 shared_ptr的函数
        • 3.3 ⾃定义类型使⽤shared_ptr
        • 3.4 vector和复制操作
        • 3.5 使⽤make_shared初始化(C++11标注)
        • 代码:

1、智能指针简介

1.1 原始指针(raw pointer)的⼀些问题

C++ 提供了内存管理的绝对⾃由度
------->• 分配
------->• 释放
------->• 声明周期管理
⼀些潜在严重问题
------->• 未初始化的指针(wild pointer),也就是野指针,可指向内存的任何位置
------->• 内存泄漏(memory leak),可能因为没有及时释放分配的内存空间
------->• 悬空指针(dangling pointer):指针指向已经释放的对象
所有权(ownership),引入智能指针
------->• 谁拥有指针?
------->• 何时可以删除指针?

1.2 智能指针(smart pointers)

• 也是对象
• 只能指向堆上分配的内存
⽤完后会⾃动删除
• 遵循RAII(资源获取即初始化)原则,对资源的申请释放,是一种成对操作的封装
• 几种C++ 智能指针:
------->• Unique pointers (unique_ptr)
------->• Shared pointers (shared_ptr)
------->• Weak pointers (weak_ptr)
------->• Auto pointers (auto_ptr) (已弃⽤)

  • 导入 #include <memory>
  • 在类模板(class templates)中定义
    • 对原始指针做了封装
    • 重载的操作符
      • 解引⽤(*)
      • 成员选择(->)
      • 不⽀持算数操作符(++, — —等)

创建智能指针的具体实例如下,smart_pointer 可以换成上面提到的 unique_ptr,shared_ptr,weak_ptr,ptr 是指向 Type 类别的智能指针,当运行完 {} 里面的程序时,会自动调用析构函数,会帮助我们处理堆上分配的内存空间,

在这里插入图片描述

2、智能指针(smart pointers)——unique_ptr

  • unique_ptr<T> ptr_name
    • 指向heap堆上类型 T 的对象
    • 唯⼀(unique),多个 unique_ptr 不可以指向同⼀个对象
    • 拥有指向对象的唯⼀所有权
    • 不可以复制或赋值,但可以移动
    • 指针使⽤完毕,被指向的对象会⾃动释放销毁
2.1 unique_ptr 的声明

在这里插入图片描述

2.2 unique_ptr 的函数

在这里插入图片描述

2.3 ⾃定义类型使⽤ unique_ptr

在这里插入图片描述

2.4 unique_ptr 不⽀持拷⻉、赋值

std::move(p1):转移p1拥有的所有权,容器 vec 拥有堆上面对象的所有权,p1 会设置为空指针,

在这里插入图片描述

2.5 使⽤make_unique初始化(C++14标准)

make_unique 的作用也是在堆上创建的内存空间,
auto 关键字是编译器根据 make_unique 的返回值自动帮我们判断数据类型,

在这里插入图片描述

代码:
#include <iostream>
#include <vector>
#include <string>
#include <memory>using namespace std;class Account
{
private:string name {"account"};double balance {0.0};
public:Account(string name = "none", double balance = 0.0);~Account();bool deposit(double amount);void printInfo() const;double getBalance();
};Account::Account(string name, double balance):name {name}, balance {balance}
{cout << "构造函数,name: " << name << endl;
}Account::~Account()
{cout << "析构函数,name: " << name << endl;
}
bool Account::deposit(double amount)
{balance += amount;return true;
}void Account::printInfo() const
{cout << "name: " << name << ", balance: " << balance << endl;
}
double Account::getBalance()
{return balance;
}int main()
{// Account alice_account {"Alice", 1000.0}; // 构造函数和析构函数都会被调用// Account * bob_account = new Account {"Bob", 2000.0}; // 只有构造函数被调用// delete bob_account; // 析构函数被调用// unique_ptr<Account> p1 {new Account {"jams", 1000.0}}; // 构造函数和析构函数都会被调用// auto p2 = make_unique<Account>("mike", 2000.0); // 构造函数和析构函数都会被调用// unique_ptr<Account> p3;// // p3 = p2; // 报错,因为unique_ptr不允许拷贝,只能移动// p3 = move(p2); // p2 会被置为null,即空指针// if (! p2)//     cout << "p2 is null" << endl;// auto p4 = make_unique<Account>("Helen", 3000.0);// p4->deposit(1000.0);// p4->printInfo(); // 调用成员函数vector<unique_ptr<Account>> accounts;accounts.push_back( make_unique<Account>("alice",1000));accounts.push_back( make_unique<Account>("bob",500));accounts.push_back( make_unique<Account>("mike",1000));for (const auto &acc: accounts) cout << acc->getBalance() << endl;return 0;
}

3、智能指针(smart pointers)——shared_ptr

  • shared_ptr<T> ptr_name
    • 指向heap堆上类型为 T 的对象
    • 不唯⼀,多个shared_ptr可以指向同⼀个对象
    • 被管理对象的所有权在多个shared_ptr中共享
    • 可以复制或赋值
    • 可以移动
    • 引⽤计数(reference count)为0,被指向的对象会⾃动释放销毁
3.1 shared_ptr的声明

当超出 {} 的作用域后,堆上的对象也会自动销毁,

在这里插入图片描述

3.2 shared_ptr的函数

use_count():返回引用计数的值,也就是当前堆上的对象被多少 shared_ptr 管理,
p1.reset() 并没有释放 p1 所指向的对象,因为 p2 还在指向这个对象,

在这里插入图片描述

3.3 ⾃定义类型使⽤shared_ptr

在这里插入图片描述

3.4 vector和复制操作

在这里插入图片描述

3.5 使⽤make_shared初始化(C++11标注)

不再使用关键字 new,编译器也可以生成更高效的执行代码,

在这里插入图片描述

代码:
#include <iostream>
#include <vector>
#include <string>
#include <memory>using namespace std;class Account
{
private:string name {"account"};double balance {0.0};
public:Account(string name = "none", double balance = 0.0);~Account();void print() const;
};Account::Account(string name, double balance):name {name}, balance {balance}
{cout << "构造函数,name: " << name << endl;
}Account::~Account()
{cout << "析构函数,name: " << name << endl;
}
void Account::print() const
{cout << "name: " << name << ", balance: " << balance << endl;
}void test_func(shared_ptr<Account> p)
{cout << "p.use_count(): " << p.use_count() << endl; // 2
}int main()
{// cout << "=====================" << endl;// shared_ptr<int> p1 {new int {100}};// cout << "p1.use_count(): " << p1.use_count() << endl; // 1// shared_ptr<int> p2 {p1}; // 共享所有权// cout << "p1.use_count(): " << p1.use_count() << endl; // 2// p1.reset(); // 释放所有权,但是不会销毁对象,因为p2还在使用// cout << "p1.use_count(): " << p1.use_count() << endl; // 0// cout << "p2.use_count(): " << p2.use_count() << endl; // 1// cout << "=====================" << endl;// shared_ptr<Account> p1 = make_shared<Account>("Alice", 1000.0);// test_func(p1);// cout << "p1.use_count(): " << p1.use_count() << endl; // 2// {//     shared_ptr<Account> p2 = p1;//     cout << "p2.use_count(): " << p2.use_count() << endl; // 3//     {//         shared_ptr<Account> p3 = p1;//         cout << "p3.use_count(): " << p3.use_count() << endl; // 4//         p1.reset();//     }//     cout << "p1.use_count(): " << p1.use_count() << endl; // 2//     cout << "p2.use_count(): " << p2.use_count() << endl; // 2// }// cout << "p1.use_count(): " << p1.use_count() << endl; // 1cout << "=====================" << endl;shared_ptr<Account> p1 = make_shared<Account>("Alice", 1000.0);shared_ptr<Account> p2 = make_shared<Account>("Bob", 2000.0);shared_ptr<Account> p3 = make_shared<Account>("Charlie", 3000.0);vector<shared_ptr<Account>> accounts;accounts.push_back(p1);accounts.push_back(p2);accounts.push_back(p3);for (const auto &p: accounts){p->print();cout << "p.use_count(): " << p.use_count() << endl; // 1}return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Go-Python-Java-C-LeetCode高分解法-第十二周合集

前言 本题解Go语言部分基于 LeetCode-Go 其他部分基于本人实践学习 个人题解GitHub连接&#xff1a;LeetCode-Go-Python-Java-C 欢迎订阅CSDN专栏&#xff0c;每日一题&#xff0c;和博主一起进步 LeetCode专栏 我搜集到了50道精选题&#xff0c;适合速成概览大部分常用算法 突…

比较Excel中的两列目录编号是否一致

使用java代码比较excel中两列是否有包含关系&#xff0c;若有包含关系&#xff0c;核对编号是否一致。 excel数据样例如下&#xff1a; package com.itownet.hg;import org.apache.poi.xssf.usermodel.XSSFSheet; import org.apache.poi.xssf.usermodel.XSSFWorkbook;import j…

C++设计模式_21_Iterator 迭代器(理解;面向对象的迭代器已过时;C++中使用泛型编程的方式实现)

Iterator 迭代器也是属于“数据结构”模式。GoF中面向对象的迭代器已经过时&#xff0c;C中目前使用泛型编程的方式实现&#xff0c;其他语言还在使用面向对象的迭代器。 文章目录 1. 动机(Motivation)2. 模式定义3. Iterator 迭代器代码分析4. 面向对象的迭代器与泛型编程实现…

基于MFC的串口通信(Mscomm)

1、串口通信的概述&#xff1a; 串口是一种重要的通信资源&#xff0c;例如鼠标口、USB接口都是串口。串行端口是CPU和串行设备间的编码转换器。当数据从CPU经过端口发送出去的时候&#xff0c;字节数据会被转为串行的位&#xff0c;在接收数据时&#xff0c;串行的位被转换为…

用Visual Studio(VS)开发UNIX/Linux项目

目录 FTP是免不了的 正确设置头文件 组织项目结构 创建何种项目类型 FTP自动上传 大部分具有Windows开发经验的程序员会比较喜欢使用Visual Studio&#xff0c;而大部分Unix/Linux程序员则喜欢使用UltraEdit直接在主机上写代码。 为什么直接在主机上写代码呢&#xff0c;因…

AIGC - Qwen大模型:Qwen-7B模型推理部署

硬件环境 作为AIGC方面的小白来说&#xff0c;我抱着非常天真的想法&#xff0c;想让它在我的工作笔记本上用i5的CPU去跑&#xff0c;至于为什么这么想&#xff0c;当然是因为我没有GPU&#xff0c;身边也没有其他的带显卡电脑 恰好&#xff0c;在腾讯云看到了GN7的显示优惠活…

内存DMA及设备内存控制详解

序言 对于PCIe 设备&#xff08;PCIe Endpoint&#xff09;来说&#xff0c;其和CPU CORE、DRAM 的交互&#xff0c;主要涉及两种类型的内存访问&#xff1a; 设备内存访问&#xff1a;PCIe 设备的 Device Memory&#xff08;设备内存&#xff09;的访问&#xff0c;例如CPU …

③ 软件工程CMM、CMMI模型【软考中级-软件设计师 考点】

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ ③ 软件工程CMM、CMMI模型【软考中级-软件设计…

flink 反压原理

背景 在flink中由于数据倾斜或者数据处理速率的不匹配&#xff0c;很容易引起反压&#xff0c;本文就看一下flink反压的原理 flink反压原理 flink全流程pineline的反压实现其实依赖于TaskManager之间的反压和TaskManager内部的反压来实现 1.TaskManager之间的反压 2.Task…

视频下载软件 Downie4 mac中文介绍

Downie mac是一款Mac平台上非常实用的视频下载工具。它支持下载各种视频网站上的视频&#xff0c;并且具有快速、稳定、易于使用的特点。 Downie支持下载各种视频网站上的视频&#xff0c;包括YouTube、Vimeo、Netflix、Hulu、Amazon等等。它具有快速、稳定的下载速度&#xff…

Python---判定表法(功能测试)

能对多条件依赖关系进行设计测试点---判定表法 等价类、边界值分析法主要关注单个输入类条件的测试 定义:是一种以表格形式表达多条件逻辑判断的工具。 条件桩: 列出问题中的所有条件&#xff0c;列出条件的次序无关紧要动作桩: 列出问题中可能采取的操作&#xff0c;操作的…

python基于VGG19实现图像风格迁移

目录 1、原理 2、代码实现 1、原理 图像风格迁移是一种将一张图片的内容与另一张图片的风格进行合成的技术。 风格&#xff08;style&#xff09;是指图像中不同空间尺度的纹理、颜色和视觉图案&#xff0c;内容&#xff08;content&#xff09;是指图像的高级宏观结构。 实…

mac 安装homebrew ,golang

mac 安装homebrew ,golang 安装homebrew安装golang选择 apple arm 版本安装配置环境变量 安装homebrew /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)"回车执行指令后&#xff0c;根据提示操作。具体包括以下提示操作&am…

每日一练 | 网络工程师软考真题Day46

阅读以下说明&#xff0c;答复以下【问题1】至【问题6】 【说明】 某公司总部效劳器1的操作系统为Windows Server 2003&#xff0c;需安装虚拟专用网〔VPN〕效劳&#xff0c;通过Internet与子公司实现平安通信&#xff0c;其网络拓扑结构和相关参数如图2-1所示。 【问题1】在Wi…

sql-50练习题16-20

sql-50练习题16-20 前言数据库表结构介绍学生表课程表成绩表教师表 1-6 检索"01"课程分数小于60&#xff0c;按分数降序排列的学生信息1-7 按平均成绩从高到低显示所有学生的所有课程的成绩以及平均成绩1-8 查询各科成绩最高分、最低分和平均分&#xff1a;以如下形式…

c++-set和map

文章目录 前言一、set容器1、set容器介绍2、set的使用2.1 set的构造函数和迭代器2.2 set的容量2.3 set修改操作 3、multiset容器3.1 multiset容器介绍3.2 multiset容器使用 二、map容器1、map容器介绍2、map容器使用2.1 map的构造函数与迭代器2.2 map中元素的修改2.3 map的容量…

Java修仙传之Flink篇

大道三千:最近我修Flink 目前个人理解&#xff1a; 处理有界&#xff0c;无界流的工具 FLINK&#xff1a; FLINK定义&#xff1a; Flink特点 Flink分层API 流的定义 有界数据流&#xff08;批处理&#xff09;&#xff1a; 有界流&#xff1a;数据结束了&#xff0c;程序也…

正则表达式包含数字和字符匹配

至少6位。 pattern : (?.[0-9])(?.[A-Za-z])[0-9A-Za-z]{6,} 正则表达式中的“?”是一个正向预查字符&#xff0c;它的意思是匹配前一个字符出现的最少一次。具体来说&#xff0c;当一个匹配出现时&#xff0c;它会检查前一个字符是否符合要求&#xff0c;如果符合&#xf…

【Java 进阶篇】深入理解 Java Response:从基础到高级

HTTP响应&#xff08;Response&#xff09;是Web开发中的一个关键概念&#xff0c;它是服务器向客户端&#xff08;通常是浏览器&#xff09;返回数据的方式。理解如何在Java中处理和构建HTTP响应是开发Web应用程序的重要一部分。本文将从基础知识到高级技巧&#xff0c;详细介…

ardupilot开发 --- 深度相机 篇

1. ZED 相机 1.1 规格 2. RealSense 需要机载计算机作为中介&#xff01;&#xff01;