答案:实现简易 shared_ptr 需定义引用计数控制块,通过构造、拷贝、赋值和析构操作管理资源;1. 使用 int* 动态记录引用计数,多个指针共享同一块内存;2. 拷贝时递增计数,析构或赋值时调用 release 函数递减,为 0 则 delete 资源;3. 支持解引用、get、use_count 等接口;4. 示例验证了构造、作用域析构、赋值等场景下引用计数正确变化;5. 关键点包括独立分配计数、自赋值保护、reset 正确释放旧资源;6. 局限性有非线程安全、无自定义删除器、未优化内存布局,生产环境需改进。

实现一个简易的 shared_ptr 智能指针,核心是模拟引用计数机制,确保多个指针共享同一块内存时,只有最后一个指针释放时才真正 delete 资源。下面是一个手动实现的基本版本,包含构造、拷贝、赋值和析构等关键操作。
我们需要一个额外的结构体来管理原始指针和引用计数,这个结构体被所有共享该对象的智能指针共用。
template <typename T>
class SharedPtr {
private:
T* ptr; // 指向实际数据的指针
int* ref_count; // 指向引用计数的指针
<pre class='brush:php;toolbar:false;'>void release() {
if (ref_count && --(*ref_count) == 0) {
delete ptr;
delete ref_count;
ptr = nullptr;
ref_count = nullptr;
}
}public: // 构造函数:接管原始指针 explicit SharedPtr(T* p = nullptr) : ptr(p), ref_count(nullptr) { if (ptr) { ref_count = new int(1); } }
// 拷贝构造函数
SharedPtr(const SharedPtr& other) : ptr(other.ptr), ref_count(other.ref_count) {
if (ref_count) {
++(*ref_count);
}
}
// 拷贝赋值运算符
SharedPtr& operator=(const SharedPtr& other) {
if (this != &other) {
release(); // 释放当前资源
ptr = other.ptr;
ref_count = other.ref_count;
if (ref_count) {
++(*ref_count);
}
}
return *this;
}
// 析构函数
~SharedPtr() {
release();
}
// 解引用
T& operator*() const { return *ptr; }
T* operator->() const { return ptr; }
// 获取原始指针
T* get() const { return ptr; }
// 获取引用计数(调试用)
int use_count() const { return ref_count ? *ref_count : 0; }
// 判断是否唯一拥有
bool unique() const { return use_count() == 1; }
// 重置指针
void reset(T* p = nullptr) {
release();
ptr = p;
if (ptr) {
ref_count = new int(1);
} else {
ref_count = nullptr;
}
}};
测试我们实现的 SharedPtr 是否正确管理引用计数。
#include <iostream>
<p>struct MyClass {
int value;
MyClass(int v) : value(v) { std::cout << "MyClass(" << v << ") created\n"; }
~MyClass() { std::cout << "MyClass destroyed\n"; }
};</p><p>int main() {
SharedPtr<MyClass> sp1(new MyClass(42));
std::cout << "use count: " << sp1.use_count() << "\n";</p><pre class='brush:php;toolbar:false;'>{
SharedPtr<MyClass> sp2 = sp1;
std::cout << "use count after copy: " << sp1.use_count() << "\n";
} // sp2 析构,引用计数减一
std::cout << "use count after sp2 destroyed: " << sp1.use_count() << "\n";
SharedPtr<MyClass> sp3;
sp3 = sp1; // 赋值测试
std::cout << "use count after assignment: " << sp1.use_count() << "\n";
return 0;}
Seede AI
AI 驱动的设计工具
713
查看详情
这个简易实现展示了 shared_ptr 的核心思想:
此实现是教学性质的简化版,生产环境的 shared_ptr 还需考虑:
基本上就这些。理解引用计数的生命周期管理,是掌握智能指针的关键。
以上就是C++如何实现一个智能指针_手动实现C++ shared_ptr的引用计数功能的详细内容,更多请关注其它相关文章!
相关文章:
拼多多视频播放卡顿如何处理 拼多多视频播放优化技巧
使用PHP从URL路径中提取倒数第二个片段
CSS子选择器:如何区分并样式化嵌套列表的子层级
高德地图怎么看全景照片_高德地图全景照片浏览教程
HTML元素状态管理:根据DIV内容动态启用/禁用按钮
快手官方唯一登录入口 谨防山寨钓鱼网站
天眼查企业查询官网入口 天眼查官方网页版查询
PDO预处理语句中冒号的正确处理:区分SQL函数格式与命名占位符
Win10文件资源管理器“此电脑”分组怎么关 Win10恢复经典视图【技巧】
QQ邮箱在线使用入口 QQ邮箱个人账号网页版登录
使用Pandas转换并合并DataFrame:多列映射至统一结构
HuggingFaceEmbeddings中向量嵌入维度调整的限制与理解
Tailwind CSS line-clamp 布局问题解析与修复指南
TikTok搜索结果不显示如何解决 TikTok搜索刷新优化方法
TypeScript/J*aScript:高效查找数组中首个唯一ID对象
Composer中的^和~符号代表什么_精通Composer版本号语义化约束
excel如何生成目录 excel一键生成工作表目录超链接
PHP表单提交消息延迟显示:Post-Redirect-Get模式深度解析与实践
深入理解字体排版:Adobe光学字偶距与CSS字偶距的差异与实现
126邮箱网页版官方入口 126邮箱账号在线登录平台
React项目中导航栏Logo自适应布局:避免裁剪与布局溢出
俄罗斯浏览器官网直达链接 俄罗斯浏览器最新在线入口导航
拷贝漫画电脑版官网入口 拷贝漫画(PC版)在线直达
J*aScript map 迭代中检测空数组元素的有效方法
使用CSS更改登录屏幕输入框中PNG图标颜色的策略与局限性
C++如何解决segmentation fault_C++段错误调试与原因分析
邮编格式怎么匹配地址_根据邮编格式快速匹配详细地址的技巧
Lar*el Form Request中唯一性验证在更新操作中的正确实现
J*aScript设计模式实践_j*ascript代码优化
PHP:从文本中提取带逗号的数字价格教程
魅族17怎样用浏览器译外语网页_iPhone魅族17浏览器译外语网页【即时翻译】
怎样把文件彻底粉碎无法恢复_Windows下安全删除敏感数据【隐私保护】
知音漫客正版漫画平台_知音漫客官网账号登录
写好的html代码怎么运行出来_运行写好的html代码方法【教程】
狙击外星人小游戏开始_狙击外星人小游戏立即开始
谷歌浏览器一键优化方案_谷歌浏览器直达主页极速不卡版
Lar*el Excel导入时生成自定义递增ID的策略与实践
Composer的 "check-platform-reqs" 命令有什么用_在部署前检查生产环境是否满足Composer依赖需求
Win11怎么关闭快速启动_Win11彻底关机设置教程
Windows 11怎么彻底关闭定位_Windows 11服务中禁用Geolocation
WooCommerce产品页高级定制:实现基于分类的交叉销售
优化HTML表单样式:解决输入框焦点跳动与元素间距问题
在Typer应用中优雅地处理和重组任意命令行参数
css子元素高度不一致导致布局错位怎么办_使用align-items:stretch解决高度差异
qq音乐在线播放入口_qq音乐电脑版登录链接
顺丰快递查单号物流信息 顺丰快递小程序查询入口
PHP面向对象编程中避免重复创建PDO数据库连接的最佳实践
《GTA6》开发画面疑似泄露!这次可不是AI了
如何创建没有密码的Windows本地账户_跳过微软账户登录的技巧【教程】
C++如何检测键盘输入_C++ _kbhit与_getch函数非阻塞输入