【Smart_Point】C/C++ 中独占指针unique_ptr
1. 独占指针unique_ptr
目录
1. 独占指针unique_ptr
1.1 unique_ptr含义
1.2 C++11特性
1.3 C++14特性
1.1 unique_ptr含义
unique_ptr 是 C++ 11 提供的用于防止内存泄漏的智能指针中的一种实现,独享被管理对象指针所有权的智能指针。unique_ptr对象包装一个原始指针,并负责其生命周期。当该对象被销毁时,会在其析构函数中删除关联的原始指针。 unique_ptr具有->和*运算符重载符,因此它可以像普通指针一样使用。 查看下面的示例:
#include <iostream>
#include <memory>
struct Task {int mId;Task(int id ) :mId(id) {std::cout << "Task::Constructor" << std::endl;}~Task() {std::cout << "Task::Destructor" << std::endl;}
};
int main()
{// 通过原始指针创建 unique_ptr 实例std::unique_ptr<Task> taskPtr(new Task(23));
//通过 unique_ptr 访问其成员int id = taskPtr->mId;std::cout << id << std::endl;
return 0;
}
对应输出:
Task::Constructor
23
Task::Destructor
unique_ptr <Task> 对象 taskPtr 接受原始指针作为参数。现在当main函数退出时,该对象超出作用范围就会调用其析构函数,在unique_ptr对象taskPtr 的析构函数中,会删除关联的原始指针,这样就不用专门delete Task对象了。 这样不管函数正常退出还是异常退出(由于某些异常),也会始终调用taskPtr的析构函数。因此,原始指针将始终被删除并防止内存泄漏。
1.2 C++11特性
1.2.1 unique_ptr 独享所有权
unique_ptr对象始终是关联的原始指针的唯一所有者。我们无法复制unique_ptr对象,它只能移动。 由于每个unique_ptr对象都是原始指针的唯一所有者,因此在其析构函数中它直接删除关联的指针,不需要任何参考计数。
创建一个空的 unique_ptr 对象
创建一个空的unique_ptr<int>对象,因为没有与之关联的原始指针,所以它是空的。
std::unique_ptr<int> ptr1;
1.2.2 检查 unique_ptr 对象是否为空
有两种方法可以检查 unique_ptr 对象是否为空或者是否有与之关联的原始指针。
// 方法1
if(!ptr1)std::cout<<"ptr1 is empty"<<std::endl;
// 方法2
if(ptr1 == nullptr)std::cout<<"ptr1 is empty"<<std::endl;
1.2.3 使用原始指针创建 unique_ptr 对象
要创建非空的 unique_ptr 对象,需要在创建对象时在其构造函数中传递原始指针,即:
std::unique_ptr<Task> taskPtr(new Task(22));
不能通过赋值的方法创建对象,下面的这句是错误的
std::unique_ptr<Task> taskPtr2 = new Task(); // 编译错误
1.3 C++14特性
1.3.1 使用 std::make_unique 创建 unique_ptr 对象 / C++14
std::make_unique<>() 是C++ 14 引入的新函数
std::unique_ptr<Task> taskPtr = std::make_unique<Task>(34);
1.3.2 获取被管理对象的指针
使用get()
·函数获取管理对象的指针。
Task *p1 = taskPtr.get();
1.3.3 重置 unique_ptr 对象
在 unique_ptr 对象上调用reset()
函数将重置它,即它将释放delete关联的原始指针并使unique_ptr 对象为空。
taskPtr.reset();
1.3.4 unique_ptr 对象不可复制
由于 unique_ptr 不可复制,只能移动。因此,我们无法通过复制构造函数或赋值运算符创建unique_ptr对象的副本。
// 编译错误 : unique_ptr 不能复制
std::unique_ptr<Task> taskPtr3 = taskPtr2; // Compile error// 编译错误 : unique_ptr 不能复制
taskPtr = taskPtr2; //compile error
1.3.5 转移 unique_ptr 对象的所有权
我们无法复制 unique_ptr 对象,但我们可以转移它们。这意味着 unique_ptr 对象可以将关联的原始指针的所有权转移到另一个 unique_ptr 对象。让我们通过一个例子来理解:
#include <iostream>
#include <memory>struct Task {int mId;Task(int id ) :mId(id) {std::cout << "Task::Constructor" << std::endl;}~Task() {std::cout << "Task::Destructor" << std::endl;}
};int main()
{// 通过原始指针创建 unique_ptr 实例std::unique_ptr<Task> taskPtr(new Task(23));//通过 unique_ptr 访问其成员int id = taskPtr->mId;std::cout << id << std::endl;// 通过原始指针创建 taskPtr2
std::unique_ptr<Task> taskPtr2(new Task(55));
// 把taskPtr2中关联指针的所有权转移给taskPtr4
std::unique_ptr<Task> taskPtr4 = std::move(taskPtr2);
// 现在taskPtr2关联的指针为空
if(taskPtr2 == nullptr)std::cout<<"taskPtr2 is empty"<<std::endl;// taskPtr2关联指针的所有权现在转移到了taskPtr4中
if(taskPtr4 != nullptr)std::cout<<"taskPtr4 is not empty"<<std::endl;// 会输出55
std::cout<< taskPtr4->mId << std::endl;return 0;
}
输出:
Task::Constructor
23
Task::Constructor
taskPtr2 is empty
taskPtr4 is not empty
55
Task::Destructor
Task::Destructor
std::move() 将把 taskPtr2
转换为一个右值引用。因此,调用 unique_ptr
的移动构造函数,并将关联的原始指针传输到 taskPtr4
。在转移完原始指针的所有权后, taskPtr2
将变为空。
1.3.6 释放关联的原始指针
在 unique_ptr
对象上调用 release()
将释放其关联的原始指针的所有权,并返回原始指针。这里是释放所有权,并没有delete原始指针,reset()
会delete原始指针。
std::unique_ptr<Task> taskPtr5(new Task(55));
// 不为空
if(taskPtr5 != nullptr)std::cout<<"taskPtr5 is not empty"<<std::endl;
// 释放关联指针的所有权
Task * ptr = taskPtr5.release();
// 现在为空
if(taskPtr5 == nullptr)std::cout<<"taskPtr5 is empty"<<std::endl;
完整示例程序
#include <iostream>
#include <memory>struct Task {int mId;Task(int id ) :mId(id) {std::cout<<"Task::Constructor"<<std::endl;}~Task() {std::cout<<"Task::Destructor"<<std::endl;}
};int main()
{// 空对象 unique_ptrstd::unique_ptr<int> ptr1;// 检查 ptr1 是否为空if(!ptr1)std::cout<<"ptr1 is empty"<<std::endl;// 检查 ptr1 是否为空if(ptr1 == nullptr)std::cout<<"ptr1 is empty"<<std::endl;// 不能通过赋值初始化unique_ptr// std::unique_ptr<Task> taskPtr2 = new Task(); // Compile Error// 通过原始指针创建 unique_ptrstd::unique_ptr<Task> taskPtr(new Task(23));// 检查 taskPtr 是否为空if(taskPtr != nullptr)std::cout<<"taskPtr is not empty"<<std::endl;// 访问 unique_ptr关联指针的成员std::cout<<taskPtr->mId<<std::endl;std::cout<<"Reset the taskPtr"<<std::endl;// 重置 unique_ptr 为空,将删除关联的原始指针taskPtr.reset();// 检查是否为空 / 检查有没有关联的原始指针if(taskPtr == nullptr)std::cout<<"taskPtr is empty"<<std::endl;// 通过原始指针创建 unique_ptrstd::unique_ptr<Task> taskPtr2(new Task(55));if(taskPtr2 != nullptr)std::cout<<"taskPtr2 is not empty"<<std::endl;// unique_ptr 对象不能复制//taskPtr = taskPtr2; //compile error//std::unique_ptr<Task> taskPtr3 = taskPtr2;{// 转移所有权(把unique_ptr中的指针转移到另一个unique_ptr中)std::unique_ptr<Task> taskPtr4 = std::move(taskPtr2);// 转移后为空if(taskPtr2 == nullptr)std::cout << "taskPtr2 is empty" << std::endl;// 转进来后非空if(taskPtr4 != nullptr)std::cout<<"taskPtr4 is not empty"<<std::endl;std::cout << taskPtr4->mId << std::endl;//taskPtr4 超出下面这个括号的作用于将delete其关联的指针}std::unique_ptr<Task> taskPtr5(new Task(66));if(taskPtr5 != nullptr)std::cout << "taskPtr5 is not empty" << std::endl;// 释放所有权Task * ptr = taskPtr5.release();if(taskPtr5 == nullptr)std::cout << "taskPtr5 is empty" << std::endl;std::cout << ptr->mId << std::endl;delete ptr;return 0;
}
输出
ptr1 is empty
ptr1 is empty
Task::Constructor
taskPtr is not empty
23
Reset the taskPtr
Task::Destructor
taskPtr is empty
Task::Constructor
taskPtr2 is not empty
taskPtr2 is empty
taskPtr4 is not empty
55
Task::Destructor
Task::Constructor
taskPtr5 is not empty
taskPtr5 is empty
66
Task::Destructor
相关文章:

ORA-01940无法删除当前已连接用户
原文地址:ORA-01940无法删除当前已连接用户作者:17361887941)查看用户的连接状况 select username,sid,serial# from v$session ------------------------------------------ 如下结果: username sid serial# ------…

距离变换扫描实现
#include <opencv2/imgproc/imgproc.hpp> #include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <iostream> // 计算欧式距离 float calcEuclideanDistance(int x1, int y1, int x2, int y2) {return sqrt(…

『扩欧简单运用』
扩展欧几里得算法 顾名思义,扩欧就是扩展欧几里得算法,那么我们先来简单地回顾一下这个经典数论算法。 对于形如\(axbyc\)的不定方程,扩展欧几里得算法可以在\(O(log_2alog_2b)\)的时间内找到该方程的一组特解,或辅助\(gcd\)判断该…

【Smart_Point】C/C++ 中共享指针 shared_ptr
1. 共享指针 shared_ptr 目录 1. 共享指针 shared_ptr 1.1 共享指针解决的问题? 1.2 创建 shared_ptr 对象 1.3 分离关联的原始指针 1.4 自定义删除器 Deleter 1.5 shared_ptr 相对于普通指针的优缺点 1.6 创建 shared_ptr 时注意事项 1.1 共享指针解决的问…

对数变换的三种实现方法
#include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <opencv2/imgproc/imgproc.hpp> #include <iostream> using namespace cv; // 对数变换方法1 cv::Mat logTransform1(cv::Mat srcImage, int c) {// 输…

eclipse编辑窗口不见了(打开左边的java、xml文件,中间不会显示代码)
转自:https://blog.csdn.net/u012062810/article/details/46729779?utm_sourceblogxgwz4 1. windows-->reset Perspective 窗口-->重置窗口布局 2. windows -> new windows 新窗口 当时手贱了一下,结果…

js的执行机制
javascript的运行机制一直困扰在我,对其的运行机制也是一知半解,在看了https://juejin.im/post/59e85eebf265da430d571f89#heading-10这篇文章后,有种茅塞顿开的感觉,下面是原文内容: 认识javascript javascript是一门单线程语言&…

【Smart_Point】unique_ptr中独占指针使用MakeFrame
1. DFrame使用方法 std::unique_ptr<deptrum::Frame> MakeFrame(deptrum::FrameType frame_type,int width,int height,int bytes_per_pixel,void* data,uint64_t timestamp,int index,float temperature) {std::unique_ptr<deptrum::Frame> frame{std::make_uniq…

最大熵阈值分割
#include <opencv2/imgproc/imgproc.hpp> #include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <iostream> using namespace std; using namespace cv; // 计算当前的位置的能量熵 float caculateCurrentE…

php登录注册
php 登录注册 注册代码:register.php <style type"text/css">form{width:300px;background-color:#EEE0E5;margin-left:300px;margin-top:30px;padding:30px;}button{margin-top:20px;} </style> <form method"post"> <la…

【C++】C/C++ 中的单例模式
目录 part 0:单例模式3种经典的实现方式 Meyers Singleton Meyers Singleton版本二 Lazy Singleton Eager Singleton Testing part 1:C之单例模式 动机 实现一[线程不安全版本] 实现二[线程安全,锁的代价过高] 锁机制 实现三[双检…

计算图像波峰点
#include "opencv2/highgui/highgui.hpp" #include "opencv2/imgproc/imgproc.hpp" #include <iostream> #include <stdio.h> #include <iostream> #include <vector> using namespace std; using namespace cv; // 计算图像的波峰…

锁的算法,隔离级别的问题
锁的算法 InnoDB存储引擎有3中行锁的算法设计,分别是: Record Lock:单个行记录上的锁。Gap Lock:间隙锁,锁定一个范围,但不包含记录本身。Next-Key Lock:Gap LockRecord Lock,锁定一…

好程序员分享24个canvas基础知识小结
好程序员分享24个canvas基础知识小结,非常全面详尽,推荐给大家。 现把canvas的知识点总结如下,以便随时查阅。 1、填充矩形 fillRect(x,y,width,height); 2、绘制矩形边框 strokeRect(x,y,width,height); 3、擦除矩形 clearRect(x,y,width,he…

【leetcode】二叉树与经典问题
文章目录笔记leetcode [114. 二叉树展开为链表](https://leetcode-cn.com/problems/flatten-binary-tree-to-linked-list/)解法一: 后序遍历、递归leetcode [226. 翻转二叉树](https://leetcode-cn.com/problems/invert-binary-tree/)思路与算法复杂度分析leetcode [剑指 Offer…

PHP PSR-1 基本代码规范(中文版)
基本代码规范 本篇规范制定了代码基本元素的相关标准,以确保共享的PHP代码间具有较高程度的技术互通性。 关键词 “必须”("MUST")、“一定不可/一定不能”("MUST NOT")、“需要”("REQUIRED")、“将会”("SHALL")、“不会…

最近邻插值实现:图像任意尺寸变换
#<opencv2/imgproc/imgproc.hpp> #include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <iostream> using namespace cv; using namespace std; // 实现最近邻插值图像缩放 cv::Mat nNeighbourInterpolatio…

软件测试-培训的套路-log3
最新的套路!我是没了解过--下图中描述-log3 Dotest-董浩 但是我知道不管什么没有白吃的午餐而且还会给钱…如果真的有,请醒醒! 当然话又回来,套路不套路,关键看你是否需要;你如果需要我觉得是帮你…不需要&…

ALSA声卡驱动中的DAPM详解之四:在驱动程序中初始化并注册widget和route
前几篇文章我们从dapm的数据结构入手,了解了代表音频控件的widget,代表连接路径的route以及用于连接两个widget的path。之前都是一些概念的讲解以及对数据结构中各个字段的说明,从本章开始,我们要从代码入手,分析dapm的…

双线性插值实现
#include <opencv2/imgproc/imgproc.hpp> #include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <iostream> // 实现双线性插值图像缩放 cv::Mat BilinearInterpolation(cv::Mat srcImage) {CV_Assert(srcI…

【C++】C++ 强制转换运算符
C 运算符 强制转换运算符是一种特殊的运算符,它把一种数据类型转换为另一种数据类型。强制转换运算符是一元运算符,它的优先级与其他一元运算符相同。 大多数的 C 编译器都支持大部分通用的强制转换运算符: (type) expression 其中&…

WPS 2019 更新版(8392)发布,搭配优麒麟 19.04 运行更奇妙!
WPS 2019 支持全新的外观界面、目录更新、方框打勾、智能填充、内置浏览器、窗口拆组、个人中心等功能。特别是全新的新建页面,让你可以整合最近打开的文档、本地模版、公文模版、在线模板等。 随着优麒麟即将发布的新版 19.04 的到来,金山办公软件也带来…

图像金字塔操作,上采样、下采样、缩放
#include "opencv2/imgproc/imgproc.hpp" #include "opencv2/highgui/highgui.hpp" #include <iostream> // 图像金子塔采样操作 void Pyramid(cv::Mat srcImage) {// 根据图像源尺寸判断是否需要缩放if(srcImage.rows > 400 && srcImage…

【C++】【OpenCv】图片加噪声处理,计时,及键盘事件响应捕捉
图像噪声添加 cv::Mat addGuassianNoise(cv::Mat& src, double a, double b) {cv::Mat temp src.clone();cv::Mat dst(src.size(), src.type()); // Construct a matrix containing Gaussian noisecv::Mat noise(temp.size(), temp.type());cv::RNG rng(time(NULL));//…

透视学理论(十四)
2019独角兽企业重金招聘Python工程师标准>>> 已知左右距点分别位于透视画面两侧,因此我们可以左距点来得出透视画面的纵深长度。 距点,指的是与透视画面成45的水平线的消失点。我们把画面边缘按1米的宽度(A点)…

适用于Mac上的SQL Server
适用于Mac上的SQL Server? 众所周知,很多人都在电脑上安装了SQL Server软件,普通用户直接去官网下载安装即可,Mac用户则该如何在Mac电脑上安装SQL Server呢?想要一款适用于Mac上的SQL Server?点击进入&…

【MATLAB】————拷贝指定文件路径下的有序文件(选择后),可处理固定规律的文件图片数据或者文件
总体上来说这种数据有2中处理思路。第一种如下所示,从一组数据中挑选出符合要求的数据; 第二中就是数据中不需要的数据删除,选择处理不需要的数据,留下的补集就是需要的数库。一般情况下需要看问题是否明确,需求明确的…

mouseover与mouseenter,mouseout与mouseleave的区别
mouseover与mouseenter 不论鼠标指针穿过被选元素或其子元素,都会触发 mouseover 事件。只有在鼠标指针穿过被选元素时,才会触发 mouseenter 事件。 mouseout与mouseleave不论鼠标指针离开被选元素还是任何子元素,都会触发 mouseout 事件。只…

图像掩码操作的两种实现
#include <opencv2/imgproc/imgproc.hpp> #include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <iostream> #include <stdio.h> using namespace cv; using namespace std; // 基于像素邻域掩码操作…

控制反转 IOC
2019独角兽企业重金招聘Python工程师标准>>> 控制反转(Inversion of Control,缩写为IoC)面向对象设计原则,降低代码耦合度 依赖注入(Dependency Injection,简称DI) 依赖查找…