C++(17)——智能指针初步及弃用auto_ptr的原因

2023-11-17

RAII

使用局部对象来管理资源的技术
在这里插入图片描述

RAII的原理

在这里插入图片描述

RAII的四个步骤

在这里插入图片描述

裸指针存在的问题:

delete后的指针变量就变成了一个失效指针(也叫作悬空指针)。

对于下面的代码:

void Destroy(Object *op)
{
	delete op;
	delete[] op;
}

Object *op = new Object(10);
Object *arop = new Object[10];

Destroy(op);
Destroy(arop);

因此:
在这里插入图片描述

智能指针

智能指针的引入:

智能指针是比原始指针更加智能的类,解决悬空指针多次删除被指向对象,以及资源泄漏问题,通常用来确保指针的寿命和其指向对象的寿命一致。
智能指针虽然很智能,很容易被误用,智能也是有代价的。

四种智能指针
  1. auto_ptr
  2. unqiue_ptr(唯一性智能指针)
  3. shared_ptr(共享性智能指针)
  4. weak_ptr(管理弱引用)

其中后三个是C11支持,并且第一个已经被C11弃用。
C98中的auto_ptr所做的事情,就是动态分配对象以及当对象不再需要时自动执行清理。

下面我们首先来了解一下为什么要将auto_ptr移除的原因:
因为该类型的智能指针意义不明确,使用浅拷贝方式时,两个对象拥有同一块资源:我们模仿源码的逻辑,了解一下:比如下面的代码:

class Object
{
    int value;
public:
    Object(int x = 0):value(x){cout<<"Create Object:"<<this<<endl;}
    ~Object(){cout<<"Destroy Object:"<<this<<endl;}

    int  & Value(){return value;}
    const int& Value() const{return value;}
};

template<class _Ty>
class my_auto_ptr
{
private:
    bool _Owns;//所有权
    _Ty* _Ptr;
public:
    my_auto_ptr(_Ty* p = NULL):_Owns(p != NULL),_Ptr(p){}
    ~my_auto_ptr()
    {
        if(_Owns)
        {
            delete _Ptr;
        }
        _Owns = false;
        _Ptr = NULL;
    }
    _Ty* get() const 
    {
        return _Ptr;
    }
    _Ty* operator->()const
    {
        return get();
    }
    _Ty & operator*()
    {
        return *get();
    }
    void reset(_Ty* p = NULL) 
    {
       if(_Owns)
       {
           delete _Ptr;
       }
       _Ptr = p;
    }
    _Ty * release()const//编译要通过,要么异变,要么强转成普通指针
    {
        _Ty* tmp = NULL;
        if(_Owns)
        {
            ((my_auto_ptr*)this)->_Owns = false;
            tmp = _Ptr;
            ((my_auto_ptr*)this)->_Ptr = NULL;
        }
        return tmp;
    }
    my_auto_ptr(const my_auto_ptr & op):_Owns(op._Owns)
    {
        if(_Owns)
        {
            _Ptr = op._Ptr;
        }
    }
};

void fun()
{
    my_auto_ptr<Object> pobj(new Object(10));//pobj是my_auto_ptr类型
    cout<<pobj->Value()<<endl;
    cout<<(*pobj).Value()<<endl;//(*pobj)是Object的堆区对象。*(pobj._Ptr).Value()
}
int main()
{
    my_auto_ptr<Object> pobja(new Object(10));
    my_auto_ptr<Object> pobjb(pobja);
}

相关函数解释:
在这里插入图片描述

此时程序必然会导致程序崩溃引发异常,主函数结束时对同一部分资源释放了两次,堆内存被释放两次
在这里插入图片描述

那么我们可能会考虑,将资源转移,即修改拷贝构造如下:利用是释放函数

my_auto_ptr(const my_auto_ptr & op):_Owns(op._Owns),_Ptr(op.release())
    {}

看似好像解决了上面的问题,实则存在隐患
在这里插入图片描述

继续来看:下面的代码存在什么问题呢?

void fun(my_auto_ptr<Object> apx)
{
    int x = apx->Value();
    cout<<x<<endl;
}

int main()
{
    my_auto_ptr<Object> pobja(new Object(10));
    
    fun(pobja);

    int a = pobja->Value();
    cout<<a<<endl;
}

上述代码的执行逻辑如下:

  1. pobja有两个域拥有权域和指针域,拿pobja初始化形参apx时,会调动拷贝构造函数
  2. apx将自己的拥有权域设为1,调动release函数,销毁了pobja对象的资源后,返回堆区对象的地址,apx接收后将自身的指针域指向原先pobja所指向的堆区对象
  3. fun函数结束,apx局部对象就会被析构,此时再打印a,对象其实已经不存在了并且自身早已失去了pobja的拥有权。

综上,此时智能指针的拷贝构造函数的两种写法:

 my_auto_ptr(const my_auto_ptr & op):_Owns(op._Owns)
    {
        if(_Owns)
        {
            _Ptr = op._Ptr;
        }
    }
   
 my_auto_ptr(const my_auto_ptr & op):_Owns(op._Owns),_Ptr(op.release())
    {}
  1. 第一种存在的问题:Object的资源会被两个释放两次
  2. 第二种存在的问题:解决了第一种问题,但是不能解决类似于实参对象初始化形参时,实参之前自身的资源丢失的问题,找不着了,因为这种情况太过于隐蔽,容易出错,所以auto_ptr作为函数参数传递时一定要避免的。或许你想到加上引用解决上面的问题,但是仔细思考后发现,我们并不知道函数对传入的传入的auto_ptr做了什么,如果当中的某些操作使其失去了对对象的所有权,那么这还可能会导致致命的执行期错误。获取再加上const 才是个不错的选择。

因此,C11标准之前的auto_ptr这个智能指针不被广泛使用的原因就是:在某些应用场景下,拷贝构造函数的意义不明确,同理赋值语句也是这个道理,意义同样不明确,因为C11标准之前并不存在移动赋值和移动构造的概念,还有就是之前谈到的一个对象和一组对象的问题,对于自定义类型而言,auto_ptr的析构函数仅能够析构一个对象,不能够处理一组对象的情况,这些都是尚未解决的问题。
于是在C11中弃用,C17标准中直接移除。
历史渊源:
在STL库之前,有一个功能更加强大的boost库,STL为了与其抗衡,应急制造了STL,但制作的不够完善,由此因为STL未解决auto_ptr的问题,因此STl内的容器vector和list都不想和auto_ptr建立联系。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++(17)——智能指针初步及弃用auto_ptr的原因 的相关文章

随机推荐

  • python wow自动打怪脚本官方教程_【按键精灵】魔兽世界LR 自动打怪脚本

    Hwnd Plugin Window Find 0 魔兽世界 设置换视角标志1 刚切换视角 0否 UserVar Flag1 0 Rem 开始 按Tab寻找目标 Call Plugin Bkgnd KeyPress Hwnd 9 Delay
  • (5)项目中DTO代替@Transient使用

    Transient 实体类字段上 此实体类和数据库表对应 某实体类对象字段要返回给前端 数据库里面没有 查询忽视数据库里面对应的字段 上面这种方法很乱也不见得好 使用DTO对象在每层数据中传输
  • Python连接MySQL

    Python连接MySQL可以使用多种库 比如pymysql mysql connector python PyMySQL等 这里以pymysql为例 介绍Python连接MySQL的基本操作 1 安装pymysql库 pip instal
  • 《ReactNative系列讲义》进阶篇---05.FlatList(二)

    版权声明 本文为博主原创文章 未经博主允许不得转载 一 简介 上篇文章中我们了解到了FlatList组件的基本用法 其实FlatList还有很多丰富的功能 可实现更多更灵活的业务需求 接下来让我们一起来看看吧 二 基础知识 支持单独的头部文
  • CUDA Samples: 获取设备属性信息

    通过调用CUDA的cudaGetDeviceProperties函数可以获得指定设备的相关信息 此函数会根据GPU显卡和CUDA版本的不同得到的结果也有所差异 下面code列出了经常用到的设备信息 include funset hpp in
  • 求助 oss异步回调 自定义参数接受不到

    客户端app服务器php获取方式 file get contents php input 获取到的值 只有系统参数
  • 网络字节序与主机字节序 高低位

    最近在项目开发过程中 需要在采用JAVA作为语言的服务器与采用C 作为语言的服务器间进行通信 这就涉及到这两种语言间数据类型的转换以及网络字节序与主机字节序的区别 该文主要说说网络字节序和主机字节序的区别以及Little endian与Bi
  • WTL 界面设计篇(CSkinComboBox)

    头文件声明 CSkinComboBox h pragma once include SkinManager h define WM CBO EDIT MOUSE HOVER WM USER 1 define WM CBO EDIT MOUS
  • 74HC595驱动7x11点阵屏(LED-7X11-JHM)DEMO

    起因 由于我之前做了一个点阵时钟 但是无奈LED点阵屏价格比较贵 所以想找一个价格较为便宜的点阵来做便宜一点的点阵方案 再淘宝上看到有那种五毛钱一个的7x11的LED点阵 所以就想着试试搞一下这种点阵屏 这个由于是7x11的点阵 没有比较好
  • Java后端项目实现无限极树 - 案例:部门树 - Department实体类

    1 domain层
  • java找不到符号解决办法

    一 java找不到符号 如果你的代码里没有报错 明明是存在的 但是java报错找不到符号 像下面这样子 二 解决步骤 1 清除编码工具缓存 本人用的idea eclipse清除缓存方式有需要的可以百度一下 2 如果是mavne项目的 先cl
  • 编程艺术 - 第一章 左旋转字符串

    题目 定义字符串的左旋转操作 把字符串前面的若干个字符移动到字符串的尾部 若把字符串abcdef左旋转2位得到字符串cdefab 请实现字符串左旋转的函数 要求对长度为n的字符串操作的时间复杂度为O n 空间复杂度为O 1 类似题目还有剑指
  • 【算法】Shell排序--C++源代码(VS2015)

    include
  • tensorflow导入错误“ImportError: DLL load failed”(已解决)

    毕业论文需要用到tensorflow 然鹅我却卡在了安装 由于各种问题还自身的拖延症与它 斗争 了一周 终于安装成功了 我一定要记录下来这血泪史 这篇笔记也拖了好几天 如果你也遇到下面的问题 就继续往下看吧 直接 pip install t
  • Docker网络理解(1)

    2017 02 17 我注意到 很多大型的企业公司在提供云计算服务的时候 必然要对各个不同的租户进行隔离 这就和OpenStack一样了 需要一个网络拓扑的设计 所以前面对网络的理解是很有用的 后续对这个隔离应用来说 我所知道的就是用OVS
  • 记一次windows下Netty做为压测端引发的错误 No buffer space available (maximum connections reached?): bind

    最近写了个客户端压测工具结果每次压到将近5000时就会报错 也是搞了两天才发现问题 主要是错误表现和网上大多数人的表现一样 导致忽略了眼前的错误提示 错误表现具体如下 java lang IllegalStateException fail
  • 兔队线段树:楼房重建

    https www luogu com cn problem P4198 本质 在线段树上每个节点维护信息时再深入到底部 加个 log log log O n
  • Promise,async,await

    什么是Promise Promise 简单说就是一个容器 里面保存着某个未来才会结束的事件 通常是一个异步操作 的结果 从语法上说 promise是一个对象 从它可以获取异步操作的的最终状态 成功或失败 Promise是一个构造函数 对外提
  • 轨迹相似性度量方法总结

    轨迹相似性度量方法总结 基于点的度量 基于形状的度量 基于分段 基于特定任务 基于点的度量 1 欧氏距离 优点 线性计算时间 缺点 轨迹长度要相同 2 DTW 是对时间序列距离测量的改进 优点 考虑到时间差 比欧式距离效果好 缺点 对噪音比
  • C++(17)——智能指针初步及弃用auto_ptr的原因

    RAII 使用局部对象来管理资源的技术 RAII的原理 RAII的四个步骤 裸指针存在的问题 delete后的指针变量就变成了一个失效指针 也叫作悬空指针 对于下面的代码 void Destroy Object op delete op d