C++ 智能指针的模拟实现实例
发布时间 - 2026-01-11 02:30:56 点击率:次C++ 智能指针的模拟实现实例

1.引入
int main()
{
int *p = new int; //裸指针
delete p;
return 0;
}
在上面的代码中定义了一个裸指针p,需要我们手动释放。如果我们一不小心忘记释放这个指针或者在释放这个指针之前,发生一些异常,会造成严重的后果(内存泄露)。而智能指针也致力于解决这种问题,使程序员专注于指针的使用而把内存管理交给智能指针。
普通指针也容易出现指针悬挂问题,当有多个指针指向同一个对象的时候,如果某一个指针delete了这个对象,所以这个指针不会对这个对象进行操作,那么其他指向这个对象的指针呢?还在等待已经被删除的基础对象并随时准备对它进行操作。于是悬垂指针就形成了,程序崩溃也“指日可待”。
int main()
{
int *p1 = new int(2);
int *p2 = p1;
int *p3 = p2;
cout<<*p1<<endl;
cout<<*p2<<endl;
cout<<*p3<<endl;
delete p1;
cout<<*p2<<endl;
return 0;
}
输出结果
2 2 2 -572662307
输出的结果*p2的结果并不是期待中2,因为2早已经被删除了。
智能指针
智能指针是一个类,它把普通指针封装起来,能实现和普通指针同样的功能。不同的是智能指针能够对内存进行自动管理,利用类对象出了作用域会调用析构函数,把对指针的释放写在析构函数中,避免出现悬挂指针的情况。
智能指针(smart pointer)是存储指向动态分配(堆)对象指针的类,用于生存期控制,能够确保自动正确的销毁动态分配的对象,防止内存泄露。它的一种通用实现技术是使用引用计数(reference count)。智能指针类将一个计数器与类指向的对象相关联,引用计数跟踪该类有多少个对象共享同一指针。每次创建类的新对象时,初始化指针并将引用计数置为1;当对象作为另一对象的副本而创建时,拷贝构造函数拷贝指针并增加与之相应的引用计数;对一个对象进行赋值时,赋值操作符减少左操作数所指对象的引用计数(如果引用计数为减至0,则删除对象),并增加右操作数所指对象的引用计数;调用析构函数时,构造函数减少引用计数(如果引用计数减至0,则删除基础对象)。
智能指针就是模拟指针动作的类。所有的智能指针都会重载 -> 和 * 操作符。智能指针还有许多其他功能,比较有用的是自动销毁。这主要是利用栈对象的有限作用域以及临时对象(有限作用域实现)析构函数释放内存。当然,智能指针还不止这些,还包括复制时可以修改源对象等。智能指针根据需求不同,设计也不同(写时复制,赋值即释放对象拥有权限、引用计数等,控制权转移等)。auto_ptr 即是一种常见的智能指针。
智能指针的实现(用类模板实现)
class Test
{
public:
Test()
{
cout<<"Test()"<<endl;
}
~Test()
{
cout<<"~Test()"<<endl;
}
void func()
{
cout<<"call Test::func()"<<endl;
}
};
template<typename T>
class CSmartptr
{
public:
CSmartptr(T *ptr):_ptr(ptr)
{cout<<"CSmartptr()"<<endl;}
CSmartptr(const CSmartptr<T> &other)
{
_ptr = new T;
*ptr = *other._ptr;
}
~CSmartptr()
{
cout<<"~CSmartptr()"<<endl;
delete _ptr;
}
void relase() const
{
((CSmartptr<T> *)this)->owns = false;
}
T& operator*()
{
return *_ptr;
}
const T& operator*()const {return *_ptr;}
T *operator->()
{
return _ptr;
}
const T *operator->()const {return _ptr;}
private:
T *_ptr;
};
int main()
{
CSmartptr<int> p1(new int);
*p1 = 200;
CSmartptr<Test> p2(new Test);
p2->func();
return 0;
}
模拟实现auto_ptr
template<typename T>
class CSmartptr
{
public:
CSmartptr(T *ptr):_ptr(ptr),owns(true){cout<<"CSmartptr()"<<endl;}
CSmartptr(const CSmartptr<T> &other)
{
other.relase();
_ptr = other._ptr;
}
~CSmartptr()
{
cout<<"~CSmartptr()"<<endl;
if( owns == true)
{
cout<<"~CSmartptr()"<<endl;
delete _ptr;
}
}
void relase() const
{
((CSmartptr<T> *)this)->owns = false;
}
T& operator*()
{
return *_ptr;
}
const T& operator*()const {return *_ptr;}
T *operator->()
{
return _ptr;
}
const T *operator->()const {return _ptr;}
private:
T *_ptr;
bool owns; //标志位 ,控制一个资源的访问权限
};
int main()
{
CSmartptr<int> p1(new int);
*p1 = 200;
CSmartptr<Test> p2(new Test);
p2->func();
return 0;
}
带有引用计数的智能指针(方便对资源的管理和释放)
class CHeapTable
{
public:
static CHeapTable& getInstance()
{
return mHeapTable;
}
//增加引用计数
void addRef(void *ptr)
{
pthread_mutex_lock(mutex);
list<Node>::iterator it = find(mList.begin(),
mList.end(), ptr); // Node == Node it->mpaddr
if(it == mList.end())
{
mList.push_front(ptr);
cout<<"new addr:"<<ptr<<" ref:"<<1<<endl;
}
else
{
it->mcount++;
cout<<"add addr:"<<ptr<<" ref:"<<it->mcount<<endl;
}
pthread_mutex_unlock(mutex);
}
//减少引用计数的
void delRef(void *ptr)
{
list<Node>::iterator it = find(mList.begin(),
mList.end(), ptr);
if(it != mList.end())
{
it->mcount--;
cout<<"del addr:"<<ptr<<" ref:"<<it->mcount<<endl;
if(it->mcount == 0)
{
mList.erase(it);
}
}
}
//获取引用计数的
int getRef(void *ptr)
{
list<Node>::iterator it = find(mList.begin(),
mList.end(), ptr);
if(it != mList.end())
{
return it->mcount;
}
return 0;
}
private:
CHeapTable(){}
static CHeapTable mHeapTable;
struct Node
{
Node(void *ptr=NULL):mpaddr(ptr),mcount(1){}
bool operator==(const Node &src)
{
return mpaddr == src.mpaddr;
}
void *mpaddr; //标识堆内存资源
int mcount; //标识资源的引用计数
};
list<Node> mList;
};
CHeapTable CHeapTable::mHeapTable;
template<typename T>
class CSmartPtr
{
public:
CSmartPtr(T *ptr = NULL)
:mptr(ptr)
{
if(mptr != NULL)
{
addRef();
}
}
~CSmartPtr()
{
delRef();
if(0 == getRef())
{
delete mptr;
mptr = NULL;
}
}
CSmartPtr(const CSmartPtr<T> &src)
:mptr(src.mptr)
{
if(mptr != NULL)
{
addRef();
}
}
CSmartPtr<T>& operator=(const CSmartPtr<T> &src)
{
if(this == &src)
return *this;
delRef();
if(0 == getRef())
{
delete mptr;
mptr = NULL;
}
mptr = src.mptr;
if(mptr != NULL)
{
addRef();
}
}
T& operator*(){return *mptr;}
const T& operator*()const{return *mptr;}
T* operator->(){return mptr;}
const T* operator->()const{return mptr;}
void addRef(){mHeapTable.addRef(mptr);}
void delRef(){mHeapTable.delRef(mptr);}
int getRef(){return mHeapTable.getRef(mptr);}
private:
T *mptr;
static CHeapTable &mHeapTable;
};
template<typename T>
CHeapTable& CSmartPtr<T>::mHeapTable = CHeapTable::getInstance();
以上就是智能指针的实例详解,如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持
# 智能指针的模拟
# 智能指针的模拟的实现
# C++智能指针实例详解
# C++中智能指针如何设计和使用
# C++ 智能指针深入解析
# C++智能指针之shared_ptr的具体使用
# C++中的智能指针举例详解及注意事项
# 的是
# 是一个
# 动态分配
# 出了
# 还在
# 多个
# 如有
# 形成了
# 指日可待
# 希望能
# 并将
# 会对
# 相关联
# 与之
# 在上面
# 即是
# 谢谢大家
# 早已经
# 写在
# 对它
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何在阿里云ECS服务器部署织梦CMS网站?
JavaScript如何操作视频_媒体API怎么控制播放
JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)
如何用AWS免费套餐快速搭建高效网站?
使用C语言编写圣诞表白程序
Claude怎样写结构化提示词_Claude结构化提示词写法【教程】
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
Laravel如何发送邮件和通知_Laravel邮件与通知系统发送步骤
Gemini怎么用新功能实时问答_Gemini实时问答使用【步骤】
Win11应用商店下载慢怎么办 Win11更改DNS提速下载【修复】
宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法
Android 常见的图片加载框架详细介绍
打造顶配客厅影院,这份100寸电视推荐名单请查收
Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南
Laravel PHP版本要求一览_Laravel各版本环境要求对照
百度浏览器网页无法复制文字怎么办 百度浏览器复制修复
Laravel如何从数据库删除数据_Laravel destroy和delete方法区别
网站制作大概多少钱一个,做一个平台网站大概多少钱?
HTML5空格和nbsp有啥关系_nbsp的作用及使用场景【说明】
想要更高端的建设网站,这些原则一定要坚持!
Laravel如何实现数据导出到CSV文件_Laravel原生流式输出大数据量CSV【方案】
微信小程序 wx.uploadFile无法上传解决办法
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法
HTML5打空格有哪些误区_新手常犯的空格使用错误【技巧】
laravel怎么使用数据库工厂(Factory)生成带有关联模型的数据_laravel Factory生成关联数据方法
浅析上传头像示例及其注意事项
Laravel观察者模式如何使用_Laravel Model Observer配置
新三国志曹操传主线渭水交兵攻略
魔毅自助建站系统:模板定制与SEO优化一键生成指南
免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?
Laravel API资源类怎么用_Laravel API Resource数据转换
php打包exe后无法访问网络共享_共享权限设置方法【教程】
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程
Laravel如何处理表单验证?(Requests代码示例)
Laravel如何使用Collections进行数据处理?(实用方法示例)
消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工
Laravel如何集成Inertia.js与Vue/React?(安装配置)
企业网站制作这些问题要关注
Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧
如何快速生成专业多端适配建站电话?
如何实现javascript表单验证_正则表达式有哪些实用技巧
高防服务器租用指南:配置选择与快速部署攻略
如何用已有域名快速搭建网站?
如何用花生壳三步快速搭建专属网站?
*服务器网站为何频现安全漏洞?
JavaScript如何实现类型判断_typeof和instanceof有什么区别
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐

