怎样实现类似智能指针的类 手写简化版智能指针教学示例

实现一个类似智能指针的类核心在于利用raii原则绑定资源与对象生命周期,通过封装原始指针确保资源自动释放,解决内存泄漏和悬空指针等问题。1. 使用模板类包装原始指针并重载解引用与成员访问运算符;2. 在析构函数中释放资源以实现自动管理;3. 禁用拷贝构造与赋值操作确保独占所有权;4. 实现移动构造与赋值转移所有权;5. 提供get、release、reset等接口用于资源获取与替换;6. 通过operator bool支持空指针检查。相比标准库unique_ptr,该简化版本缺少自定义删除器、数组特化、make_unique支持及多态转换等功能,但已体现raii与移动语义的核心机制。

怎样实现类似智能指针的类 手写简化版智能指针教学示例

实现一个类似智能指针的类,核心在于利用C++的RAII(Resource Acquisition Is Initialization)原则,将资源的生命周期管理与对象的生命周期绑定。简单来说,就是用一个类来包装原始指针,确保当这个包装类对象被销毁时,它所持有的资源(比如动态分配的内存)也能被自动释放,从而避免内存泄漏和悬空指针等问题。

怎样实现类似智能指针的类 手写简化版智能指针教学示例

我一直觉得,理解智能指针,光看概念是远远不够的,得自己上手写一个,哪怕是简化版。你会发现,那些平时觉得有点玄乎的RAII原则,一下就变得具体起来了。

怎样实现类似智能指针的类 手写简化版智能指针教学示例

#include  // 仅用于示例中的输出// 示例:一个简单的资源类,用于观察构造和析构struct MyResource {    int value;    MyResource(int v) : value(v) {        std::cout << "MyResource(" << value << ") constructed.n";    }    ~MyResource() {        std::cout << "MyResource(" << value << ") destroyed.n";    }    void do_something() {        std::cout << "MyResource(" << value << ") doing something.n";    }};// 简化版智能指针:MyUniquePtr// 模仿std::unique_ptr,实现独占所有权template class MyUniquePtr {private:    T* ptr; // 持有的原始指针public:    // 构造函数:接受一个原始指针    // explicit关键字避免隐式类型转换    explicit MyUniquePtr(T* p = nullptr) : ptr(p) {        // std::cout << "MyUniquePtr constructed with ptr: " << ptr << "n";    }    // 析构函数:确保在MyUniquePtr对象销毁时,其管理的资源也被释放    ~MyUniquePtr() noexcept { // 析构函数应为noexcept        // std::cout << "MyUniquePtr destructed, deleting ptr: " << ptr << "n";        delete ptr;    }    // 禁用拷贝构造函数和拷贝赋值运算符    // 独占所有权意味着不能复制,只能移动    MyUniquePtr(const MyUniquePtr&) = delete;    MyUniquePtr& operator=(const MyUniquePtr&) = delete;    // 移动构造函数:从另一个MyUniquePtr对象“窃取”所有权    MyUniquePtr(MyUniquePtr&& other) noexcept : ptr(other.ptr) {        other.ptr = nullptr; // 将原对象置空,防止其析构时误删资源        // std::cout << "MyUniquePtr move constructed.n";    }    // 移动赋值运算符:同理,转移所有权    MyUniquePtr& operator=(MyUniquePtr&& other) noexcept {        if (this != &other) { // 避免自我赋值            delete ptr; // 先释放当前持有的资源            ptr = other.ptr;            other.ptr = nullptr;        }        // std::cout <访问成员    T* operator->() const {        return ptr;    }    // 获取原始指针:通常不建议直接使用,除非必要    T* get() const noexcept {        return ptr;    }    // 释放所有权:返回原始指针,并将MyUniquePtr置空    // 调用者现在负责管理返回的原始指针    T* release() noexcept {        T* oldPtr = ptr;        ptr = nullptr;        return oldPtr;    }    // 重置:释放当前资源,并管理新的原始指针    void reset(T* p = nullptr) noexcept {        if (ptr != p) { // 避免删除自身(如果p就是当前ptr)            delete ptr;            ptr = p;        }    }    // 转换为bool:判断是否持有有效指针    explicit operator bool() const noexcept {        return ptr != nullptr;    }};// 示例用法// int main() {//     std::cout << "--- Creating p1 ---n";//     MyUniquePtr p1(new MyResource(10)); // 独占MyResource(10)//     p1->do_something();//     std::cout << "p1 value: " << (*p1).value << "n";//     std::cout << "--- Attempting copy (will fail to compile) ---n";//     // MyUniquePtr p2 = p1; // 编译错误:拷贝构造函数被禁用//     std::cout << "--- Moving p1 to p3 ---n";//     MyUniquePtr p3 = std::move(p1); // 移动所有权,p1变空//     if (p1) {//         std::cout << "p1 is still valid (should not happen).n";//     } else {//         std::cout <do_something();//     std::cout << "p3 value: " <value << "n";//     std::cout <do_something();//     std::cout << "--- Releasing p3's ownership ---n";//     MyResource* rawPtr = p3.release(); // p3放弃所有权,MyResource(20)未被销毁//     if (!p3) {//         std::cout << "p3 is now empty after release.n";//     }//     std::cout << "Manually deleting released rawPtr...n";//     delete rawPtr; // 必须手动删除,否则MyResource(20)会泄漏//     std::cout << "--- End of main ---n";//     return 0;// }

为什么我们需要智能指针?它解决了哪些传统C++内存管理痛点?

说实话,刚开始写C++那会儿,内存泄漏简直是家常便饭。每次程序崩溃,都得花大量时间去排查是不是哪个

new

没配对

delete

。智能指针简直就是救星,它把这种繁琐、易错的活儿自动化了。

传统C++内存管理中,我们经常会遇到几个让人头疼的问题。最常见的就是内存泄漏,当你

new

了一块内存,却因为各种原因(比如忘记

delete

、提前返回、异常抛出)没有释放它,这块内存就永远被占用了,直到程序结束。想象一下一个长时间运行的服务,如果频繁发生内存泄漏,最终会导致系统资源耗尽。

怎样实现类似智能指针的类 手写简化版智能指针教学示例

另一个痛点是悬空指针和重复释放。当你

delete

了一块内存后,如果原始指针没有被置空,它就成了悬空指针。之后如果再通过这个悬空指针去访问内存,或者再次

delete

它,就会导致程序崩溃或者未定义行为。这在复杂的程序中,尤其是有多个指针指向同一块内存时,更是噩梦。

此外,异常安全也是个大问题。如果在函数内部

new

了内存,然后因为某些操作抛出了异常,而

delete

语句在异常点之后,那么内存就永远不会被释放了。智能指针通过将资源管理封装在类的析构函数中,确保了无论函数如何退出(正常返回还是抛出异常),析构函数都会被调用,从而保证了资源被正确释放,这就是C++中非常重要的RAII(Resource Acquisition Is Initialization)原则。它不仅仅限于内存,文件句柄、网络连接、锁等任何需要在特定时刻获取和释放的资源,都可以通过RAII原则来管理。

实现一个简化的智能指针时,有哪些关键设计考量?

设计这东西,很多时候都是在权衡。比如,我们这个简化版,为了突出核心的RAII和所有权,就没去考虑多线程下的引用计数(那是

shared_ptr

的范畴),也没去搞自定义删除器。但即便是这样,里面的移动语义,还有对拷贝的禁用,都是缺一不可的。

实现一个简化版智能指针,主要有以下几个关键设计考量:

所有权语义(Ownership Semantics):这是智能指针的核心。我们选择实现的是独占所有权(

unique_ptr

风格),这意味着任何时候,只有一个智能指针实例拥有对特定资源的控制权。为了强制这种独占性,我们必须禁用拷贝构造函数和拷贝赋值运算符。如果允许拷贝,就会出现多个智能指针管理同一块内存的情况,导致重复释放。

RAII原则的体现:这是自动管理资源的关键。智能指针的构造函数负责获取资源(接收一个原始指针),而析构函数则负责释放资源(调用

delete

)。当智能指针对象超出作用域或被销毁时,其析构函数会自动被调用,从而保证资源被及时、安全地释放。

操作符重载:为了让智能指针的行为尽可能接近原始指针,我们需要重载

*

(解引用运算符)和

->

(成员访问运算符)。这样,用户就可以像使用原始指针一样,通过智能指针来访问所指向对象的值或成员。

移动语义(Move Semantics):虽然我们禁用了拷贝,但为了实现所有权的转移(比如从一个函数返回智能指针,或者将智能指针放入容器),移动构造函数和移动赋值运算符是必不可少的。移动操作会将资源的所有权从一个智能指针转移到另一个,同时将原智能指针置空,避免资源被多次管理。

空指针处理:智能指针应该能够安全地处理空指针。例如,当智能指针不持有任何资源时(即内部的

ptr

nullptr

),对其进行

reset()

操作或析构时,不应该导致问题。同时,提供一个

operator bool()

重载可以方便地检查智能指针是否持有有效资源。

release()

reset()

方法

release()

允许智能指针放弃对资源的控制权,并返回原始指针,这在某些需要手动管理资源或将资源传递给C风格API的场景下非常有用。

reset()

则允许智能指针释放当前持有的资源,并开始管理一个新的资源。

我们的简化版智能指针与标准库中的

std::unique_ptr

有何异同?

写完这个简化版,你会发现,标准库里的

std::unique_ptr

真的考虑得太周全了。我们这个版本,就像是个“毛坯房”,能住人,但很多细节功能还没完善。比如自定义删除器,那真是个特别实用的功能,能让智能指针管理各种非内存资源。但话说回来,能搭起这个“毛坯房”,就已经很能说明问题了。

我们的

MyUniquePtr

与C++标准库中的

std::unique_ptr

在核心思想和功能上是相似的,都实现了独占所有权和RAII原则,并支持移动语义。然而,标准库的版本经过了大量的工程实践和优化,拥有更多高级特性和健壮性:

相同点:

独占所有权: 两者都确保了资源在任何时候只被一个智能指针实例拥有。RAII原则: 都通过构造函数获取资源,通过析构函数释放资源,保证了资源管理的自动化和异常安全。移动语义: 都支持通过移动操作来转移所有权,而非复制。操作符重载: 都重载了

*

->

,使得它们可以像原始指针一样使用。

get()

release()

reset()

这些核心接口功能基本一致。

不同点(

MyUniquePtr

的局限性):

自定义删除器(Custom Deleters):

std::unique_ptr

允许你指定一个自定义的删除器(可以是函数对象、lambda表达式等),来处理资源的释放。这使得

unique_ptr

不仅可以管理堆内存,还能管理文件句柄(

FILE*

)、网络套接字等需要特定释放操作的资源。我们的

MyUniquePtr

目前只能使用

delete

操作符。数组支持:

std::unique_ptr

有针对数组的特化版本(

std::unique_ptr

),它的析构函数会调用

delete[]

来正确释放数组内存。而我们的

MyUniquePtr

只调用

delete ptr;

,如果用于管理数组,会导致未定义行为。

make_unique

函数: 标准库提供了

std::make_unique

辅助函数,用于创建

unique_ptr

对象。它不仅语法更简洁,还能提供异常安全保证(避免在

new

unique_ptr

构造之间发生异常)。类型转换和多态:

std::unique_ptr

支持从派生类指针到基类指针的安全转换,这在多态场景下非常有用。我们的简化版可能没有完全实现这一点。

noexcept

保证:

std::unique_ptr

的许多成员函数都带有

noexcept

关键字,表明它们不会抛出异常,这对于编写异常安全的代码非常重要。我在

MyUniquePtr

中也尝试添加了,但在实际的生产级代码中,这需要更细致的

以上就是怎样实现类似智能指针的类 手写简化版智能指针教学示例的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1469541.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 18:02:18
下一篇 2025年12月18日 18:02:27

相关推荐

  • CSS mask属性无法获取图片:为什么我的图片不见了?

    CSS mask属性无法获取图片 在使用CSS mask属性时,可能会遇到无法获取指定照片的情况。这个问题通常表现为: 网络面板中没有请求图片:尽管CSS代码中指定了图片地址,但网络面板中却找不到图片的请求记录。 问题原因: 此问题的可能原因是浏览器的兼容性问题。某些较旧版本的浏览器可能不支持CSS…

    2025年12月24日
    900
  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 为什么设置 `overflow: hidden` 会导致 `inline-block` 元素错位?

    overflow 导致 inline-block 元素错位解析 当多个 inline-block 元素并列排列时,可能会出现错位显示的问题。这通常是由于其中一个元素设置了 overflow 属性引起的。 问题现象 在不设置 overflow 属性时,元素按预期显示在同一水平线上: 不设置 overf…

    2025年12月24日 好文分享
    400
  • 网页使用本地字体:为什么 CSS 代码中明明指定了“荆南麦圆体”,页面却仍然显示“微软雅黑”?

    网页中使用本地字体 本文将解答如何将本地安装字体应用到网页中,避免使用 src 属性直接引入字体文件。 问题: 想要在网页上使用已安装的“荆南麦圆体”字体,但 css 代码中将其置于第一位的“font-family”属性,页面仍显示“微软雅黑”字体。 立即学习“前端免费学习笔记(深入)”; 答案: …

    2025年12月24日
    000
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么我的特定 DIV 在 Edge 浏览器中无法显示?

    特定 DIV 无法显示:用户代理样式表的困扰 当你在 Edge 浏览器中打开项目中的某个 div 时,却发现它无法正常显示,仔细检查样式后,发现是由用户代理样式表中的 display none 引起的。但你疑问的是,为什么会出现这样的样式表,而且只针对特定的 div? 背后的原因 用户代理样式表是由…

    2025年12月24日
    200
  • inline-block元素错位了,是为什么?

    inline-block元素错位背后的原因 inline-block元素是一种特殊类型的块级元素,它可以与其他元素行内排列。但是,在某些情况下,inline-block元素可能会出现错位显示的问题。 错位的原因 当inline-block元素设置了overflow:hidden属性时,它会影响元素的…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 为什么使用 inline-block 元素时会错位?

    inline-block 元素错位成因剖析 在使用 inline-block 元素时,可能会遇到它们错位显示的问题。如代码 demo 所示,当设置了 overflow 属性时,a 标签就会错位下沉,而未设置时却不会。 问题根源: overflow:hidden 属性影响了 inline-block …

    2025年12月24日
    000
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 为什么我的 CSS 元素放大效果无法正常生效?

    css 设置元素放大效果的疑问解答 原提问者在尝试给元素添加 10em 字体大小和过渡效果后,未能在进入页面时看到放大效果。探究发现,原提问者将 CSS 代码直接写在页面中,导致放大效果无法触发。 解决办法如下: 将 CSS 样式写在一个单独的文件中,并使用 标签引入该样式文件。这个操作与原提问者观…

    2025年12月24日
    000
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 em 和 transition 设置后元素没有放大?

    元素设置 em 和 transition 后不放大 一个 youtube 视频中展示了设置 em 和 transition 的元素在页面加载后会放大,但同样的代码在提问者电脑上没有达到预期效果。 可能原因: 问题在于 css 代码的位置。在视频中,css 被放置在单独的文件中并通过 link 标签引…

    2025年12月24日
    100
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信