C++:独占指针(unique_ptr)的理解

2024-08-28 17:12
文章标签 c++ 指针 理解 unique ptr 独占

本文主要是介绍C++:独占指针(unique_ptr)的理解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引入

在C++中,动态内存的管理是通过运算符new/delete来完成的:

  • new:在动态内存中为对象分配空间,并且返回一个指向该对象的指针,我们可以选择返回对象对其进行初始化;
  • delete:接受一个动态对象的指针,销毁该对象,并且释放与之关联的内存。

动态分配的对象的生命周期与它们在哪里创建是无关的,只有当显式地被释放时,这些对象才会销毁。

当我们对动态内存的使用不当时,会出现很多麻烦:

  1. 有些内存资源已经被释放,但指向它的指针并没有改变指向(成为了野指针),并且后续还在使用;
  2. 有些内存资源已经被释放,后期又试图再释放一次(重复释放同一块内存会导致程序运行崩溃);
  3. 没有及时释放不再使用的内存资源,造成内存泄漏,程序占用的内存资源越来越多。

为了更容易同时也更安全的使用动态内存,C++11提供了智能指针来管理动态对象。智能指针的行为类似于常规指针,重要的区别在于它负责自动释放所指向的对象。

原理

当我们创建一个类对象时,会自动调用类的默认构造函数。当类对象超出作用域时,会自动调用析构函数来释放资源。

智能指针的实现原理是通过RAII(Resource Acquisition Is Initialization,资源获取即初始化)技术来管理动态分配的内存。它通过在对象的构造函数中获取资源,在对象的析构函数中释放资源,来保证资源的正确使用。

unique_ptr

unique_ptr是独占指针,在同一时刻他只允许一个指针指向对象。

std::unique_ptr<int> u1(new int(2));
auto u2=std::make_unique<int>(2);

由于unique_ptr的独占性,不允许对unique_ptr进行拷贝和赋值。

unique_ptr(const unique_ptr&)=delete;
unique_ptr& operator=(const unique_ptr&)=delete;

可以利用std::move将对象所有权从一个unique_ptr转移给另一个unique_ptr。转移后,原来的unique_ptr将不再拥有对内存的控制权,将变为空指针。

//1.使用release来转移
std::unique_ptr<Test> u1=std::make_unique<Test>(11);
std::unique_ptr<Test> u2(u1.release());//2.使用move来转移
std::unique_ptr<Test> u1=std::make_unique<Test>(11);
std::unique_ptr<Test> u2(std::move(u1));

由于unique_ptr不能被拷贝,所以把unique_ptr作为参数类型会报错

void testUniquePtr(std::unique_ptr<Test> &ptr) {}

如果函数的参数不是引用类型,并且外部不再需要使用该指针,我们可以使用move()或者release()把该对象的控制权转移,将其交由调用的函数管理

void testUniquePtr(std::unique_ptr<Test> ptr) {} //作用域结束,将会释放ptr所指向的对象int main(){std::unique_ptr<Test> up = std::make_unique<Test>(100);//将对象的唯一控制权交给了函数//函数结束后,对象被释放testUniquePtr(std::unique_ptr<Test>(std::move(up)));return 0;
}

如果想把把unique_ptr作为参数返回时,可以调用其move()方法,同时还可以把返回的unique_ptr转换为shared_ptr

//使用move操作,而非拷贝构造函数
std::unique_ptr<Test> test(int i) {return std::make_unique<Test>(i);
}int main(){//使用move给up赋值,而非拷贝构造函数std::unique_ptr<Test> up = test(100);//可以把一个对象的控制权交由shared_ptr来管理std::shared_ptr<Test> sp = test(100);return 0;
}

自定义删除器

unique_ptr的模板类型为:

template<typename __Tp,typename __Dp=default_delete<__Tp> >
class unique_ptr{
......
};

模板的参数,第一个为unique_ptr关联的原始指针类型,后者为删除器,默认值为default_delete、删除器是unique_ptr类型的组成部分,可以是普通函数指针、函数对象或者lambda表达式。当需要自定义删除器时,我们需要指定其类型,即__Dp不可省略,我们可以通过decltype来获得其类型。

//1.函数指针
void myDeleter(Test* p){delete p;
}std::unique_ptr<Test,decltype<&myDeleter) > ptr1(new Test(),myDeleter);//2.函数对象
class MyDeleter{
public:void operator()(int *p){delete[] p;}
};
std::unique_ptr<int[],MyDeleter> p2(new int[100],MyDeleter());//3.lambda表达式std::unique_ptr<int,void(*)(int*)> p3(new int(1),[](int *p){ delete p;});

常用函数

get():返回智能指针中保存的原生指针

reset():释放原生指针,并将原指针置空

reset(q):释放原生指针,使智能指针指向新的原生指针。

swap(p,q):交换智能指针p和q的原生指针

独占指针unique_ptr不需要维护引用计数和原子操作,因此比共享指针shared_ptr所消耗的内存更少,性能也更好。

template<class T>
class Unique_Ptr{
private:T* m_p;
public:explicit Unique_Ptr(T* p=nullptr):m_p(p) {}~Unique_Ptr(){if(m_p)delete m_p;}Unique_Ptr(const Unique_Ptr& )=delete;Unique_Ptr& operator=(const Unique_Ptr&)=delete;//移动构造函数Unique_Ptr(Unique_Ptr&& that) noexcept :m_p(that.m_p){that.m_p=nullptr;}//移动赋值函数Unique_Ptr& operator=(Unique_Ptr&& that) noexcept {if(this!=&that){if(m_p!=nullptr){delete m_p;}m_p=that.m_p;that.m_p=nullptr;}return *this;}void swap(Unique_Ptr& that) noexcept {std::swap(m_p,that.m_p);}T* get() const noexcept {return m_p;}T* release() noexcept {T* tmp=m_p;m_p=nullptr;return tmp;}void reset() noexcept {if(m_p!=nullptr)delete m_p;m_p=nullptr;      }void reset(T* p) noexcept {if(m_p!=nullptr)delete m_p;m_p=p;}T& operator*() noexcept {return *m_p;}T* operator->() noexcept {return m_p;}explicit operator bool() const noexcept {return m_p != nullptr;}bool operator==(Unique_Ptr const& that) const noexcept {return m_p == that.m_p;}bool operator!=(Unique_Ptr const& that) const noexcept {return m_p != that.m_p;}};

这篇关于C++:独占指针(unique_ptr)的理解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115383

相关文章

一文带你理解Python中import机制与importlib的妙用

《一文带你理解Python中import机制与importlib的妙用》在Python编程的世界里,import语句是开发者最常用的工具之一,它就像一把钥匙,打开了通往各种功能和库的大门,下面就跟随小... 目录一、python import机制概述1.1 import语句的基本用法1.2 模块缓存机制1.

深入理解C语言的void*

《深入理解C语言的void*》本文主要介绍了C语言的void*,包括它的任意性、编译器对void*的类型检查以及需要显式类型转换的规则,具有一定的参考价值,感兴趣的可以了解一下... 目录一、void* 的类型任意性二、编译器对 void* 的类型检查三、需要显式类型转换占用的字节四、总结一、void* 的

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

深入理解Redis大key的危害及解决方案

《深入理解Redis大key的危害及解决方案》本文主要介绍了深入理解Redis大key的危害及解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 目录一、背景二、什么是大key三、大key评价标准四、大key 产生的原因与场景五、大key影响与危

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数