文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C++之智能指针初步及弃用auto_ptr的原因是什么

2023-07-05 16:00

关注

本篇内容介绍了“C++之智能指针初步及弃用auto_ptr的原因是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

RAII

使用局部对象来管理资源的技术

C++之智能指针初步及弃用auto_ptr的原因是什么

RAII的四个步骤

C++之智能指针初步及弃用auto_ptr的原因是什么

裸指针存在的问题

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

对于下面的代码:

void Destroy(Object *op){delete op;delete[] op;}Object *op = new Object(10);Object *arop = new Object[10];Destroy(op);Destroy(arop);

因此:

C++之智能指针初步及弃用auto_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);}

相关函数解释:

C++之智能指针初步及弃用auto_ptr的原因是什么

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

C++之智能指针初步及弃用auto_ptr的原因是什么

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

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

看似好像解决了上面的问题,实则存在隐患

C++之智能指针初步及弃用auto_ptr的原因是什么

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

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;}

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

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

 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())    {}

因此,C11标准之前的auto_ptr这个智能指针不被广泛使用的原因就是:在某些应用场景下,拷贝构造函数的意义不明确,同理赋值语句也是这个道理,意义同样不明确,因为C11标准之前并不存在移动赋值和移动构造的概念,还有就是之前谈到的一个对象和一组对象的问题,对于自定义类型而言,auto_ptr的析构函数仅能够析构一个对象,不能够处理一组对象的情况,这些都是尚未解决的问题。

于是在C11中弃用,C17标准中直接移除。 

历史渊源:

在STL库之前,有一个功能更加强大的boost库,STL为了与其抗衡,应急制造了STL,但制作的不够完善,由此因为STL未解决auto_ptr的问题,因此STl内的容器vector和list都不想和auto_ptr建立联系。

“C++之智能指针初步及弃用auto_ptr的原因是什么”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯