【C++11及其特性】智能指针——unique_ptr

2024-09-02 20:04
文章标签 c++ 指针 特性 智能 unique ptr

本文主要是介绍【C++11及其特性】智能指针——unique_ptr,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

unique_ptr目录

  • 一.排他所有权模式
  • 二.auto_ptr的缺点
    • 1.可以直接复制和拷贝构造
    • 2.STL可以直接赋值
    • 3.不支持动态内存分配数组
  • 三.unique_ptr(C++11)
    • 1.不支持直接赋值和构造
    • 2.STL可以不可以直接赋值
    • 3.支持动态内存分配数组
  • 四.unique_ptr的用法
    • 1.构造函数
    • 2.赋值操作
    • 3.主动释放对象
    • 4.放弃对象控制权
    • 5.重置
    • 6.交换
  • 五.排他性智能指针的陷阱
  • 六.完整代码
  • 七.总结

一.排他所有权模式

在这里插入图片描述
运行结果:
在这里插入图片描述
auto_ptr和unique_ptr都是排他所有权模式,就是一块内存只能一个指针来管理.

所以这个过程是p1先释放,然后被p2赋值,p2就释放了.

二.auto_ptr的缺点

1.可以直接复制和拷贝构造

在这里插入图片描述
这里就是可以直接赋值,但是其真正的跟我们的赋值又存在着差异.
不符合习惯.赋值后p2是空的.

2.STL可以直接赋值

在这里插入图片描述
在这里插入图片描述

原理和刚刚的赋值其实是一样的

3.不支持动态内存分配数组

auto_ptr只支持单个的动态内存分配.
在这里插入图片描述

三.unique_ptr(C++11)

1.不支持直接赋值和构造

unique_ptr则不允许直接赋值和构造,因为排他性.
这样也符合我们的习惯.
在这里插入图片描述
如果硬要转的话,可以用move移动语义,将左值变为右值.
在这里插入图片描述

2.STL可以不可以直接赋值

在这里插入图片描述

3.支持动态内存分配数组

在这里插入图片描述

四.unique_ptr的用法

以下将会用Test类来测试.
在这里插入图片描述

1.构造函数

  • 没有指向值的.
    在这里插入图片描述
  • 直接初始化的.
    在这里插入图片描述
  • 数组的.
    在这里插入图片描述
  • 带删除器的,就是生命周期结束的话,调用的自定义的,可能要在释放前做点什么.
    在这里插入图片描述
    在这里插入图片描述
    运行结果:
    在这里插入图片描述

2.赋值操作

  • 通过reset()赋值
    在这里插入图片描述
  • 通过move语义赋值
    在这里插入图片描述

3.主动释放对象

不用等生命周期结束.
在这里插入图片描述

4.放弃对象控制权

不管理其释放了.
在这里插入图片描述

5.重置

重置指向的指针.
在这里插入图片描述

6.交换

交换指向的指针.
在这里插入图片描述

五.排他性智能指针的陷阱

排他所有权模式,虽然让我们无法两个指针指向同一块内存.
但是当使用reset时,可以使其两个指针指向同一块内存了.
如果一个生命周期结束,那么下一个生命周期结束的话又会进行释放,导致重复释放.
在这里插入图片描述

六.完整代码

#include <iostream>
#include <memory>
#include <vector>
using namespace std;int main1()
{auto_ptr<string>p1(new string("我是p1"));auto_ptr<string>p2(new string("我是p2"));cout << *p1 << endl;cout << *p2 << endl;cout << p1.get() << endl;cout << p2.get() << endl;p1 = p2;//排他所有权模式cout << *p1 << endl;//cout << *p2 << endl;//为空,不能访问cout << p1.get() << endl;cout << p2.get() << endl;cout << endl;vector<auto_ptr<string>>va;auto_ptr<string>p3(new string("我是p3"));auto_ptr<string>p4(new string("我是p4"));va.push_back(move(p3));va.push_back(move(p4));cout << va[0].get() << endl;cout << va[1].get() << endl;va[0] = va[1];cout << va[0].get() << endl;cout << va[1].get() << endl;cout << *va[0] << endl;//cout << *va[1] << endl;//不能访问为空//auto_ptr<int []>ai(new int[5]);//陷阱auto_ptr<string>p5;auto_ptr<string>p6;string* str = new string("智能指针的内存管理陷阱");p5.reset(str);{p6.reset(str);}cout << p6.get() << endl;cout << p5.get() << endl;system("pause");return 0;
}int main2()
{unique_ptr<string>p1(new string("我是p1"));unique_ptr<string>p2(new string("我是p2"));cout << *p1 << endl;cout << *p2 << endl;cout << p1.get() << endl;cout << p2.get() << endl;//p1 = p2;//直接报错不允许//unique_ptr<string>p7(p1);//拷贝构造也不可以unique_ptr<string>p7(move(p1));p1 = move(p2);//只有只有才可以赋值cout << *p1 << endl;//cout << *p2 << endl;cout << p1.get() << endl;cout << p2.get() << endl;cout << endl;vector<unique_ptr<string>>va;unique_ptr<string>p3(new string("我是p3"));unique_ptr<string>p4(new string("我是p4"));va.push_back(move(p3));va.push_back(move(p4));cout << va[0].get() << endl;cout << va[1].get() << endl;//va[0] = va[1];//不允许赋值cout << va[0].get() << endl;cout << va[1].get() << endl;cout << *va[0] << endl;//cout << *va[1] << endl;//auto_ptr<int []>ai(new int[5]);unique_ptr<int[]>ui(new int[5]);//支持//陷阱auto_ptr<string>p5;auto_ptr<string>p6;string* str = new string("智能指针的内存管理陷阱");p5.reset(str);{p6.reset(str);}cout << p6.get() << endl;cout << p5.get() << endl;system("pause");return 0;
}class Test
{
public:Test(){cout << "调用构造函数" << endl;}~Test(){cout << "调用析构函数" << endl;}void doSomething(){cout << "do...." << endl;}
};class DestrucTest
{
public:void operator()(Test* p){p->doSomething();delete p;}
};int main()
{//赋值unique_ptr<Test>up1;Test* t = new Test();up1.reset(t);unique_ptr<Test>up2(new Test());unique_ptr<Test>up3;up3 = move(up2);up3.release();unique_ptr<Test[]>up4(new Test[5]);up4 = NULL;//主动释放,up4.reset()也可以{unique_ptr<Test, DestrucTest>up5(new Test());}unique_ptr<Test>up6(new Test());unique_ptr<Test>up7(new Test());up6.swap(up7);system("pause");return 0;
}

七.总结

用排他性的智能指针时,就用独一无二的unique_ptr.
auto_ptr已经被淘汰了!

这篇关于【C++11及其特性】智能指针——unique_ptr的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

C/C++ chrono简单使用场景示例详解

《C/C++chrono简单使用场景示例详解》:本文主要介绍C/C++chrono简单使用场景示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友... 目录chrono使用场景举例1 输出格式化字符串chrono使用场景China编程举例1 输出格式化字符串示

Java空指针异常NullPointerException的原因与解决方案

《Java空指针异常NullPointerException的原因与解决方案》在Java开发中,NullPointerException(空指针异常)是最常见的运行时异常之一,通常发生在程序尝试访问或... 目录一、空指针异常产生的原因1. 变量未初始化2. 对象引用被显式置为null3. 方法返回null

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以

C++类和对象之默认成员函数的使用解读

《C++类和对象之默认成员函数的使用解读》:本文主要介绍C++类和对象之默认成员函数的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、默认成员函数有哪些二、各默认成员函数详解默认构造函数析构函数拷贝构造函数拷贝赋值运算符三、默认成员函数的注意事项总结一

C/C++中OpenCV 矩阵运算的实现

《C/C++中OpenCV矩阵运算的实现》本文主要介绍了C/C++中OpenCV矩阵运算的实现,包括基本算术运算(标量与矩阵)、矩阵乘法、转置、逆矩阵、行列式、迹、范数等操作,感兴趣的可以了解一下... 目录矩阵的创建与初始化创建矩阵访问矩阵元素基本的算术运算 ➕➖✖️➗矩阵与标量运算矩阵与矩阵运算 (逐元

C/C++的OpenCV 进行图像梯度提取的几种实现

《C/C++的OpenCV进行图像梯度提取的几种实现》本文主要介绍了C/C++的OpenCV进行图像梯度提取的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录预www.chinasem.cn备知识1. 图像加载与预处理2. Sobel 算子计算 X 和 Y

C/C++和OpenCV实现调用摄像头

《C/C++和OpenCV实现调用摄像头》本文主要介绍了C/C++和OpenCV实现调用摄像头,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录准备工作1. 打开摄像头2. 读取视频帧3. 显示视频帧4. 释放资源5. 获取和设置摄像头属性

c/c++的opencv图像金字塔缩放实现

《c/c++的opencv图像金字塔缩放实现》本文主要介绍了c/c++的opencv图像金字塔缩放实现,通过对原始图像进行连续的下采样或上采样操作,生成一系列不同分辨率的图像,具有一定的参考价值,感兴... 目录图像金字塔简介图像下采样 (cv::pyrDown)图像上采样 (cv::pyrUp)C++ O

c/c++的opencv实现图片膨胀

《c/c++的opencv实现图片膨胀》图像膨胀是形态学操作,通过结构元素扩张亮区填充孔洞、连接断开部分、加粗物体,OpenCV的cv::dilate函数实现该操作,本文就来介绍一下opencv图片... 目录什么是图像膨胀?结构元素 (KerChina编程nel)OpenCV 中的 cv::dilate() 函