【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ]

这篇具有很好参考价值的文章主要介绍了【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ]。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ],C++,c++,开发语言

引言

在 C++ 中,智能指针是一种非常重要的概念,它能够帮助我们自动管理动态分配的内存,避免出现内存泄漏等问题。在上一篇文章中,我们了解了智能指针的基本概念和原理,本篇文章将继续介绍 auto_ptrunique_ptr 两种智能指针的概念及其在 C++ 中的模拟实现。通过学习这些内容,您将更好地理解智能指针的不同类型和使用场景,进一步提高程序的安全性和可靠性。让我们一起探索C++智能指针的精彩世界!

一、std::auto_ptr

🔴std::auto_ptr官方文档
【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ],C++,c++,开发语言

1. 简介

std::auto_ptr 是 C++ 标准库中提供的一种智能指针类型,用于管理动态分配的内存资源。

std::auto_ptr 的主要特点是拥有所有权语义的转移。当一个 auto_ptr 对象拥有某个内存资源时,它可以将这个所有权转移给另一个 auto_ptr 对象。这意味着当一个 auto_ptr 对象被赋值给另一个 auto_ptr 对象或者被传递给一个函数时,原来的 auto_ptr 对象将不再拥有该资源,而是转移到了新的对象上

std::auto_ptr 类的定义在 <memory> 头文件中。使用 auto_ptr 需要包含该头文件,并使用 std 命名空间。

2. 使用示例

✅我们可以使用 auto_ptr 来管理动态分配的整型对象:

#include <iostream>
#include <memory>

int main() {
    std::auto_ptr<int> ptr(new int(42));
    
    std::cout << *ptr << std::endl;  // 输出:42
    
    std::auto_ptr<int> ptr2 = ptr;  // 所有权转移
    
    std::cout << *ptr2 << std::endl;  // 输出:42
    // std::cout << *ptr << std::endl;  // 错误!ptr 不再拥有资源

    return 0;
}

🚨🚨注意当一个 auto_ptr 对象转移所有权后,原来的对象将变为一个空指针。这可能导致程序出现意外的行为,因此需要谨慎使用。此外,std::auto_ptr 并不支持数组类型的内存资源管理,它只适用于单个对象。如果需要管理动态分配的数组,应该使用其他智能指针类型,如 std::unique_ptrstd::shared_ptr

总之,std::auto_ptr 是 C++ 标准库中提供的一种智能指针类型,具有所有权转移的特性。然而,由于其存在一些潜在的问题,已经在 C++17 中被废弃,推荐使用更先进的智能指针类型来代替

3. C++模拟实现

template<class T>
class auto_ptr
{
public:
    // 构造函数,接受一个指向动态分配的资源的指针
    auto_ptr(T* ptr)
        : _ptr(ptr)
    {}

    // 析构函数,在对象销毁时释放资源
    ~auto_ptr()
    {
        if (_ptr)
        {
            cout << "delete:" << _ptr << endl;
            delete _ptr;
        }
    }

    // 拷贝构造函数,用于管理权转移
    auto_ptr(auto_ptr<T>& ap)
        : _ptr(ap._ptr)
    {
        ap._ptr = nullptr;
    }

    // 解引用操作符,返回所管理资源的引用
    T& operator*()
    {
        return *_ptr;
    }

    // 成员访问操作符,返回所管理资源的指针
    T* operator->()
    {
        return _ptr;
    }
private:
    T* _ptr;  // 指向动态分配的资源的指针
};

这段代码是一个简化版的 auto_ptr 类的实现,用于演示 auto_ptr 的基本工作原理。它是一个模板类,可以管理任意类型的动态分配的内存资源。

⭕该类包含了以下成员函数和成员变量:

  • 构造函数:接受一个指向动态分配的资源的指针,并将其存储在私有成员变量 _ptr 中。
  • 析构函数:在对象销毁时释放资源,即调用 delete 操作符删除 _ptr 指向的内存。
  • 拷贝构造函数:用于管理权转移,将另一个 auto_ptr 对象的资源转移到当前对象中,并将原对象的指针置为空。
  • operator*:用于解引用 auto_ptr 对象,返回所管理资源的引用。
  • operator->:用于访问 auto_ptr 所管理资源的成员。

🚨🚨再次提醒当一个 auto_ptr 对象转移所有权后,原来的对象将变为一个空指针。这可能导致程序出现意外的行为,因此需要谨慎使用

二、std::unique_ptr

🔴std::unique_ptr官方文档
【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ],C++,c++,开发语言

1. 简介

std::unique_ptr 是 C++11 中引入的一种智能指针,它是一个轻量级的、不可拷贝的指针类型。与传统的裸指针不同,std::unique_ptr 通过 RAII 的方式来管理动态分配的内存资源,从而实现自动资源释放和防止内存泄漏

使用 std::unique_ptr 可以避免手动管理动态分配的内存资源,因为 std::unique_ptr 自身就拥有资源的所有权,当 std::unique_ptr 被销毁时,它所管理的资源也会被自动释放。这使得代码更加简洁、安全和易于维护

std::unique_ptr 有以下几个主要特点:

  1. 不支持拷贝和赋值操作,即不能直接复制或赋值一个 std::unique_ptr 对象,只能通过移动语义或者 std::move 函数来转移资源的所有权;
  2. 在默认情况下,std::unique_ptr 使用 delete 操作符来释放所管理的资源,但也可以通过自定义删除器来实现对资源的自定义释放操作;
  3. 支持使用 lambda 表达式来实现自定义删除器,从而更加灵活地管理资源。

2. 使用示例

下面是一个简单的示例,演示了 std::unique_ptr 的基本使用方法:

#include <iostream>
#include <memory>

int main()
{
    // 使用 std::unique_ptr 来管理动态分配的 int 类型对象
    std::unique_ptr<int> uptr(new int(42));

    // 解引用操作符,返回所管理资源的引用
    std::cout << *uptr << std::endl;

    // 成员访问操作符,返回所管理资源的指针
    int* p = uptr.get();
    std::cout << *p << std::endl;

    // 试图复制或赋值 unique_ptr 对象会编译错误
    // std::unique_ptr<int> uptr2 = uptr; // Error

    // 转移资源所有权
    std::unique_ptr<int> uptr2 = std::move(uptr);
    std::cout << *uptr2 << std::endl;

    return 0;
}

输出结果为

42
42
42

🚨🚨注意:由于 std::unique_ptr 是一个模板类,可以管理任意类型的动态分配的内存资源,因此使用时需要显式指定模板参数。另外,对于数组的动态分配内存资源的管理,建议使用 std::unique_ptr 的数组版本 std::unique_ptr<T[]>

3. C++模拟实现

template<class T>
class unique_ptr
{
public:
    // 构造函数,接受一个裸指针作为参数
    unique_ptr(T* ptr)
        :_ptr(ptr)
    {}

    // 析构函数,释放资源
    ~unique_ptr()
    {
        if (_ptr)
        {
            cout << "delete:" << _ptr << endl;
            delete _ptr;
        }
    }

    // 重载解引用操作符,返回资源的引用
    T& operator*()
    {
        return *_ptr;
    }

    // 重载成员访问操作符,返回资源的指针
    T* operator->()
    {
        return _ptr;
    }

    // 防止拷贝和赋值
    // C++11思路:直接将拷贝构造函数和赋值运算符声明为删除函数
    unique_ptr(const unique_ptr<T>& up) = delete;
    unique_ptr<T>& operator=(const unique_ptr<T>& up) = delete;

    // 防止拷贝和赋值
    // C++98思路:只声明不实现,但是用的人可能会在外部强行定义,所以再加一条,声明为私有
    // private:
    // unique_ptr(const unique_ptr<T>& up);
    // unique_ptr<T>& operator=(const unique_ptr<T>& up);

private:
    T* _ptr;
};

🚩这段代码实现了一个简化版的 unique_ptr 类,具有管理动态资源的能力,并防止了拷贝和赋值操作。注释中详细解释了每个函数的作用和实现原理,以及两种防止拷贝和赋值的方式(C++11 和 C++98 思路)

温馨提示

感谢您对博主文章的关注与支持!另外,我计划在未来的更新中持续探讨与本文相关的内容,会为您带来更多关于C++以及编程技术问题的深入解析、应用案例和趣味玩法等。请继续关注博主的更新,不要错过任何精彩内容!

再次感谢您的支持和关注。期待与您建立更紧密的互动,共同探索C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!
【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ],C++,c++,开发语言文章来源地址https://www.toymoban.com/news/detail-804154.html

到了这里,关于【C++入门到精通】智能指针 auto_ptr、unique_ptr简介及C++模拟实现 [ C++入门 ]的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C++11中的智能指针unique_ptr、shared_ptr和weak_ptr详解

    目录 1、引言 2、什么是智能指针? 3、在Visual Studio中查看智能指针的源码实现 4、独占式指针unique_ptr 4.1、查看unique_ptr的源码实现片段 4.2、为什么unique_ptr的拷贝构造函数和复制函数被delete了?(面试题) 4.3、使用unique_ptr独占式智能指针的实例 5、共享式指针shared_ptr  5.1、查

    2024年02月08日
    浏览(29)
  • C++11 新特性 ⑥ | 智能指针 unique_ptr、shared_ptr 和 weak_ptr

    目录 1、引言 2、unique_ptr 3、shared_ptr 4、weak_ptr 5、shared_ptr循环引用问题(面试题)

    2024年02月09日
    浏览(30)
  • C++11补充:智能指针如std::unique_ptr如何添加自定义的deleter

    关于智能指针的问题,有时候为了方便,需要在析构时附加一些清理操作,或者,有的指针本身来自C的库文件中,这时候,就非常期望能够使用自定义的deleter, 但是标准C++创建函数如std::make_unique / std::make_shared 等不支持创建带有自定义deleter的智能指针,这时,我们只能使用

    2024年02月07日
    浏览(29)
  • 【C++入门到精通】智能指针 [ C++入门 ]

    在C++编程中,内存管理一直是一个重要的话题。手动分配和释放内存可能会导致各种问题,例如内存泄漏和悬挂指针,这些问题往往会导致程序崩溃或产生不可预测的结果。为了解决这些问题, C++提供了一种称为智能指针的机制,它可以自动管理内存分配和释放,从而避免了

    2024年01月21日
    浏览(31)
  • C++智能指针shared_ptr用法

    写在前面的总结: 一个shared_ptr对象管理一个指针(new T,在堆空间),多个shared_ptr对象可以管理同一个指针,只有某个shared_ptr对象第一次初始化指针时才执行指针的构造函数,管理同一个指针的shared_ptr对象个数称为引用计数,这个引用计数保存在每个管理该指针的shared_p

    2024年02月03日
    浏览(46)
  • C++智能指针shared_ptr详解

    C++智能指针shared_ptr是一种可以自动管理内存的智能指针,它是C++11新增的特性之一。与传统指针不同,shared_ptr可以自动释放所管理的动态分配对象的内存,并避免了手动释放内存的繁琐操作,从而减少了内存泄漏和野指针的出现。 shared_ptr是一个模板类,通过引用计数器实现

    2023年04月22日
    浏览(28)
  • C++智能指针shared_ptr使用详解

    shared_ptr 是一个共享所有权的智能指针,允许多个指针指向同一个对象。 ​ shared_ptr 使用 引用计数 ,每一个shared_ptr的拷贝都指向相同的内存。每使用它一次,内部的引用计数加1,每析构一次,内部的引用计数减1,减为0时,释放所指向的堆内存。shared_ptr内部的引用计数是安

    2024年02月07日
    浏览(30)
  • C++之weak_ptr与shared_ptr智能指针实例(一百九十五)

    简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏: Audio工程师进阶系列 【 原创干货持续更新中…… 】🚀 人生格言: 人生从来没有捷径,只有行动才是治疗恐惧和懒惰的唯一良药. 更多原创,欢迎关注:An

    2024年02月09日
    浏览(48)
  • C++从入门到精通——auto的使用

    C++的auto用于自动推导变量的类型,让编译器根据变量的初始化表达式来确定其类型。使用auto可以简化代码,并且可以在某些情况下提高代码的可读性和灵活性。 使用auto声明变量时,变量的类型将根据初始化表达式的类型进行推导。例如: auto也可以和引用一起使用,

    2024年04月25日
    浏览(17)
  • shared_ptr和unique_ptr主动释放

    shared_ptr和unique_ptr均可以采用reset()来进行释放,unique_ptr调用了reset之后就会直接释放掉,shared_ptr则会在所有引用计数变为0的时候才会释放申请的内存。 注意unique_ptr的release()方法,并不会释放资源,只会把unique_ptr置为空指针,原来那个资源可以继续调用 reset 结果: 在reset之

    2024年02月14日
    浏览(26)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包