0

0

如何在C++中实现引用计数智能指针 手写简化版shared_ptr示例

P粉602998670

P粉602998670

发布时间:2025-07-10 14:24:02

|

187人浏览过

|

来源于php中文网

原创

实现一个简化版的c++++引用计数智能指针(类似shared_ptr)的核心步骤包括:设计包含资源指针和引用计数的类,实现构造函数、拷贝构造函数、析构函数和赋值操作符重载,以及提供获取原始指针和解引用的操作符。2. 构造函数初始化资源并设置初始引用计数为1;拷贝构造函数使新对象共享资源并递增引用计数;析构函数递减引用计数并在其变为0时释放资源;赋值操作符处理自我赋值并正确释放旧资源后接管新资源。3. 通过重载*和->操作符以支持指针访问语义,并提供use_count方法查看当前引用计数值。4. 示例代码演示了该智能指针的基本用法及引用计数的变化过程,验证了其核心机制的有效性。

如何在C++中实现引用计数智能指针 手写简化版shared_ptr示例

在C++中,实现一个简化版的引用计数智能指针(类似shared_ptr)可以帮助你理解其底层机制。下面会一步步带你写一个基础版本,虽然不考虑线程安全、自定义删除器等复杂情况,但足以展示核心原理。

如何在C++中实现引用计数智能指针 手写简化版shared_ptr示例

基本结构设计

要实现一个带有引用计数的智能指针,我们需要两个关键部分:

如何在C++中实现引用计数智能指针 手写简化版shared_ptr示例
  • 一个资源管理类,用来持有实际的对象指针;
  • 一个引用计数管理类,记录当前有多少个智能指针指向这个资源。

我们可以用一个结构体来保存这些信息:

立即学习C++免费学习笔记(深入)”;

template
class RefCountedPtr {
private:
    struct ControlBlock {
        T* ptr;
        int* ref_count;

        ControlBlock(T* p) : ptr(p), ref_count(new int(1)) {}
        ~ControlBlock() {
            delete ptr;
            delete ref_count;
        }
    };

    ControlBlock control_block;
};

当然这只是骨架,后面我们会把它封装成更像shared_ptr的形式。

如何在C++中实现引用计数智能指针 手写简化版shared_ptr示例

实现构造与析构逻辑

我们要让多个智能指针共享同一个资源,并在最后一个指针被销毁时释放内存。

Bertha.ai
Bertha.ai

一款专为WordPress打造的AI内容和图像创建工具

下载

构造函数

构造函数负责初始化资源和引用计数:

template
class SimpleSharedPtr {
private:
    T* ptr;
    int* ref_count;

public:
    explicit SimpleSharedPtr(T* p = nullptr)
        : ptr(p), ref_count(new int(1)) {}

    // ...
};

拷贝构造函数

拷贝构造函数需要让新对象指向同一资源,并增加引用计数:

SimpleSharedPtr(const SimpleSharedPtr& other)
    : ptr(other.ptr), ref_count(other.ref_count) {
    ++(*ref_count);
}

析构函数

当引用计数变为0时才真正释放资源:

~SimpleSharedPtr() {
    if (--(*ref_count) == 0) {
        delete ptr;
        delete ref_count;
    }
}

赋值操作符重载

赋值操作要考虑自我赋值和旧资源释放的问题:

SimpleSharedPtr& operator=(const SimpleSharedPtr& other) {
    if (this != &other) {
        // 先减少当前引用计数
        if (--(*ref_count) == 0) {
            delete ptr;
            delete ref_count;
        }

        // 接管对方资源
        ptr = other.ptr;
        ref_count = other.ref_count;
        ++(*ref_count);
    }
    return *this;
}

获取原始指针和解引用操作

为了让我们的智能指针像普通指针一样使用,还需要重载一些操作符:

T& operator*() const { return *ptr; }
T* operator->() const { return ptr; }

T* get() const { return ptr; }

示例代码完整版

下面是简化版shared_ptr的一个最小可用实现:

#include 

template
class SimpleSharedPtr {
private:
    T* ptr;
    int* ref_count;

public:
    explicit SimpleSharedPtr(T* p = nullptr)
        : ptr(p), ref_count(new int(1)) {}

    SimpleSharedPtr(const SimpleSharedPtr& other)
        : ptr(other.ptr), ref_count(other.ref_count) {
        ++(*ref_count);
    }

    SimpleSharedPtr& operator=(const SimpleSharedPtr& other) {
        if (this != &other) {
            if (--(*ref_count) == 0) {
                delete ptr;
                delete ref_count;
            }

            ptr = other.ptr;
            ref_count = other.ref_count;
            ++(*ref_count);
        }
        return *this;
    }

    ~SimpleSharedPtr() {
        if (--(*ref_count) == 0) {
            delete ptr;
            delete ref_count;
        }
    }

    T& operator*() const { return *ptr; }
    T* operator->() const { return ptr; }

    int use_count() const { return *ref_count; }
};

int main() {
    SimpleSharedPtr p1(new int(42));
    std::cout << "Use count: " << p1.use_count() << std::endl; // 输出 1

    {
        SimpleSharedPtr p2 = p1;
        std::cout << "Use count: " << p1.use_count() << std::endl; // 输出 2
    }

    std::cout << "Use count after p2 destroyed: " << p1.use_count() << std::endl; // 输出 1
    return 0;
}

基本上就这些了。
这个版本没有考虑线程安全、自定义删除器、数组支持等功能,但在学习理解引用计数机制上已经足够。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

185

2025.07.04

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

467

2023.08.10

excel制作动态图表教程
excel制作动态图表教程

本专题整合了excel制作动态图表相关教程,阅读专题下面的文章了解更多详细教程。

24

2025.12.29

freeok看剧入口合集
freeok看剧入口合集

本专题整合了freeok看剧入口网址,阅读下面的文章了解更多网址。

74

2025.12.29

俄罗斯搜索引擎Yandex最新官方入口网址
俄罗斯搜索引擎Yandex最新官方入口网址

Yandex官方入口网址是https://yandex.com;用户可通过网页端直连或移动端浏览器直接访问,无需登录即可使用搜索、图片、新闻、地图等全部基础功能,并支持多语种检索与静态资源精准筛选。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

207

2025.12.29

python中def的用法大全
python中def的用法大全

def关键字用于在Python中定义函数。其基本语法包括函数名、参数列表、文档字符串和返回值。使用def可以定义无参数、单参数、多参数、默认参数和可变参数的函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

16

2025.12.29

python改成中文版教程大全
python改成中文版教程大全

Python界面可通过以下方法改为中文版:修改系统语言环境:更改系统语言为“中文(简体)”。使用 IDE 修改:在 PyCharm 等 IDE 中更改语言设置为“中文”。使用 IDLE 修改:在 IDLE 中修改语言为“Chinese”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

18

2025.12.29

C++的Top K问题怎么解决
C++的Top K问题怎么解决

TopK问题可通过优先队列、partial_sort和nth_element解决:优先队列维护大小为K的堆,适合流式数据;partial_sort对前K个元素排序,适用于需有序结果且K较小的场景;nth_element基于快速选择,平均时间复杂度O(n),效率最高但不保证前K内部有序。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

12

2025.12.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号