C++ 智能指针 boost::scoped_ptr分析
1.scoped_ptr的实现原理及特性
特性:scoped_ptr和auto_ptr类似,但最大的区别就是不能转让管理权限,也就是说scoped_ptr禁止用户进行拷贝和赋值
实现原理:如何才能禁止一个类进行拷贝和复制呢?我们只需要将类的拷贝构造函数和赋值运算符重载的访问限定符设置为私有的可以
样例如下:
class ScopedPtr{
private:
ScopedPtr(const ScopedPtr& sp);
ScopedPtr& operator(const ScopedPtr& sp);
};
scoped_ptr和auto_ptr类似,它包装了new操作符在堆上分配的动态对象,能够保证动态创建的对象在任何时候都可以被正确的删除,但是scoped_ptr的所有权更加严格,不能转让,一旦scoped_ptr获得了对象的管理权,你就无法再从它那里去回来
正如scoped_ptr(局部指针)名字的含义一样:这个指针只能在作用域里使用,不希望被转让
实现如下:
template<class T>
class scoped_ptr
{
private:
T *px;
scoped_ptr(scoped_ptr const &);
scoped_ptr & operator=(scoped_ptr const &);
public:
explicit scoped_ptr(T *p = 0);
~scoped_ptr(); void reset(T *p = 0); T & operator*()const;
T * operator->()const;
T * get()const; operator unspecified-bool-type()const;
void swap(scoped_ptr & b);
};
分析:
scoped_ptr的构造函数接受一个类型为T*的指针p,创建出一个scoped_ptr对象,并在内部保存指针参数p,p必须是一个new表达式动态分配的结果或是一个空指针,当scoped_ptr的对象生命周期结束时,析构函数会使用delete操作自动销毁所保存的指针对象,从而正确的回收资源。
scoped_ptr同时把拷贝构造函数和赋值操作都声明为私有,禁止对scoped_ptr的复制操作,保证了被它管理的指针不能被转让所有权
成员函数reset的功能是重置scoped_ptr,它删除原来保存的指针,再保存新的指针值p,如果p是空指针,那么scopted_ptr将不能持有任何指针,一般情况下reset不应该配调用,因为它违背了scopted_ptr的本意---资源应该一直由scoped-ptr自己自动管理
实际上拥有权不可转移不够方便,swap成员函数可以交换两个scopted_ptr保存的原始指针,需要知道的是,scoped1.swap(scoped2) 只能用于它的定义所在的智能指针,而swap(scoped1,scoped2) 可以更广泛的用于很多指针类型,包括裸指针和第三方智能指针
scoped_ptr用operator*()和operator->()重载了引用操作符和箭头操作符,以模仿被代理的原始指针的行为,因此可以把scoped_ptr对象如同指针一样使用,如果scoped_ptr保存空指针,那么这两个操作都是未定义的
scoped_ptr不支持比较操作,不能在两个scoped_ptr之间,scoped_ptr和原始指针之间,scoped_ptr和空指针之间,进行相等或者不相等的比较操作,我们也无法为它编写额外的比较函数,因为其=和!=两个操作符都是私有的
2.scoped_ptr与auto_ptr的区别
1.scoped_ptr和auto_ptr的用法几乎一样,大多数情况下都可以与auto_ptr互换,它可以从一个auto_ptr获得指针的管理权(同时auto_ptr失去指针管理权)
2.scoped_ptr和auto_ptr一样不能用作容器的元素,但是原因不同,auto_ptr是因为它的转移语义,而scoped_ptr则是不支持拷贝和赋值,不符合容器对元素类型的要求
3.scoped_ptr和auto_ptr的根本区别在于所有权,auto_ptr被特意设计为所有权是可以被转移的,可以在函数之间传递,同一时刻只能有一个auto_ptr管理指针,而scoped_ptr把拷贝构造函数和赋值函数都声明为私有的,拒绝了指针所有权的转让,只有scoped_ptr自己能够管理指针,其他人都无权访问被管理的指针,从而保证了指针的绝对安全
4.如果代码企图从一个scoped_ptr构造或赋值另一个scoped_ptr,那么编译器会报错,阻止你这么做,从而保护你的代码,scoped_ptr更明确的表达了原始代码编写者的意图:只能在定义的作用域内使用,不可转让,这在代码后续的维持生命周期中很重要
样例如下:
template<typename T>
class ScopedPtr
{
public:
ScopedPtr(T* ptr=NULL):_ptr(ptr){}
~ScopedPtr(){
if(_ptr!=NULL)
{
delete _ptr;
_ptr=NULL;
}
}
private:
ScopedPtr(const ScopedPtr &sp);
ScopedPtr& operator=(const ScopedPtr &sp);
private:
T *_ptr;
}; int main()
{
ScopedPtr<int> sp1(new int(10));
ScopedPtr<int> sp2(new int(20)); //ScopedPtr<int> sp3(sp1); //编译错误 拷贝函数私有
//sp1=sp2; //编译错误 =操作符私有
}
由于boost::scoped_ptr独享指针所有权,当我们真的需要复制时,需求便满足不了,如此我们再引入一个智能指针boost::shared_ptr专门处理复制,参数传递的情况,下一节我们探讨boost::shared_ptr智能指针
C++ 智能指针 boost::scoped_ptr分析的更多相关文章
- [4] 智能指针boost::scoped_ptr
[1]boost::scoped_ptr简介 boost::scoped_ptr属于boost库,定义在namespace boost中,包含头文件#include <boost/scoped_ ...
- [5] 智能指针boost::shared_ptr
[1]boost::shared_ptr简介 boost::shared_ptr属于boost库,定义在namespace boost中,包含头文件#include<boost/shared_p ...
- [6] 智能指针boost::weak_ptr
[1]boost::weak_ptr简介 boost::weak_ptr属于boost库,定义在namespace boost中,包含头文件 #include<boost/weak_ptr.hp ...
- 详解 boost 库智能指针(scoped_ptr<T> 、shared_ptr<T> 、weak_ptr<T> 源码分析)
一.boost 智能指针 智能指针是利用RAII(Resource Acquisition Is Initialization:资源获取即初始化)来管理资源.关于RAII的讨论可以参考前面的文章.在使 ...
- 【C/C++学院】0904-boost智能指针/boost多线程锁定/哈希库/正則表達式
boost_array_bind_fun_ref Array.cpp #include<boost/array.hpp> #include <iostream> #includ ...
- 关于智能指针boost::shared_ptr
boost库中的智能指针shared_ptr, 功能强大, 且开销小,故受到广大coder的欢迎. 但在实际的使用过程中,笔者也发现了一些不足. 1.定制的删除器 shared_ptr除了可以使用默认 ...
- C++ 智能指针 std::auto_ptr 分析
背景介绍: RAll机制 定义一个类来封装资源的分配和释放,在构造函数中完成资源的分配和初始化,在析构函数中完成资源的清理,从而保证资源的正确初始化和清理 ps:智能指针就是RAll机制的一种应用,智 ...
- Qt智能指针简明说明
下面的智能指针分别对应boost库,Qt库,c++11的智能指针 boost::scoped_ptr QScopedPointer unique_ptr 在其生命期结束后会自动删除它所指的对象(确定 ...
- Boost中的智能指针(转)
这篇文章主要介绍 boost中的智能指针的使用.(转自:http://www.cnblogs.com/sld666666/archive/2010/12/16/1908265.html) 内存管理是一 ...
随机推荐
- cube.js 通过presto-gateway 进行连接
cube.js 对于presto 的支持是通过presto-client 刚好简单修改了一个可以支持presto-gateway 连接的 以下是一个简单的集成,以及关于集成中原有的一些修改 环境准备 ...
- singer tap-minio-csv 使用
使用tap-minio-csv 我们可以将s3 中csv 的文件,通过singer 的target 写到不用的系统中,可以兼容 s3 的存储类型,以下是一个集成minio 的测试,将minio 中的c ...
- cogs 944. [東方S3] 藤原妹红
二次联通门 : cogs 944. [東方S3] 藤原妹红 /* cogs 944. [東方S3] 藤原妹红 最小生成树 + 树形dp 首先对原图跑最下生成树 后建出一棵树 在树上进行dp 先走到叶子 ...
- socket数据传输
目录 subprocess模块 struct模块: 粘包问题: QQ聊天的实现: 文件的传输: 大文件的传输: 传输层协议: TCP : UDP: FTP: socketServer模块: subpr ...
- Golang 接口
1 接口是什么 Golang中没有像Python.Java拥有类和对象的概念,其封装对象或说明对象是通过接口来实现的.比如谁能够实现什么样的功能,便能够将其抽象化封装. 接口定义了一组方法(抽象方法集 ...
- linux 搭建局域网YUM源仓库服务器
yum简介 Yum(全称为 Yellow dog Updater, Modified)是一个在Fedora和RedHat以及CentOS中的Shell前端软件包管理器.基于RPM包管理,能够从指定的服 ...
- jmeter(五十一)_性能测试中的服务器资源监控与分析
概述 性能测试过程中,对服务器资源的监控是必不可少的.这里的资源又分了两块,windows和linux linux下监控资源 访问网址http://jmeter-plugins.org/downl ...
- QPS/TPS的预估
先说标准概念: TPS:Transactions Per Second(每秒传输的事物处理个数),即服务器每秒处理的事务数.TPS包括一条消息入和一条消息出,加上一次用户数据库访问.(业务TPS = ...
- Gradle 教程: 教你配置全局国内仓库,解决新建项目卡顿,下载构建慢等问题
想必各位从Maven 转过来的大佬们,对于maven中配置国内仓库的方法还记忆深刻.通过/用户目录下/.m2/settings.xml 局部配置或者修改全局配置.不过没有接触过maven 也不要紧,可 ...
- 【转】PostgreSQL与MySQL比较
转自:https://www.cnblogs.com/geekmao/p/8541817.html PostgreSQL与MySQL比较 特性 MySQL PostgreSQL 实例 通过执行 M ...