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) 内存管理是一 ...
随机推荐
- Git git2.8.1客户端安装教程
Git git2.8.1客户端安装教程 一 Git介绍 git是一款免费.开源的分布式版本控制系统,用于敏捷高效地处理任何或小或大的项目. 分布式相比于集中式的最大区别在于开发者可以提交到本地,每个开 ...
- MongoDB 启动报错
1.配置MongoDB ls /etc/mongod.conf 可以根据此配置文件启动 或者根据自己需求进行配置文件的变更 重要提醒: 如果变更MongoDB配置文件中:日志与数据文件目录,那么要把这 ...
- BZOJ 4903: [Ctsc2017]吉夫特 数论+dp
思路很巧妙的一道题 ~ 这个应该不完全是正解,复杂度约为 $O(3\times 10^8)$,有时间再研究研究正解. 首先,最裸的暴力是按照权值从小到大枚举每一个数,然后枚举后面的数来更新方案数,是 ...
- Getting A Mime Type From A File Name In .NET Core
Getting a mime type based on a file name (Or file extension), is one of those weird things you never ...
- P1501 [国家集训队]Tree II LCT
链接 luogu 思路 简单题 代码 #include <bits/stdc++.h> #define ls c[x][0] #define rs c[x][1] using namesp ...
- VS设置自动压缩CSS
- 5098: [BZOJ1098][POI2007]办公楼biu
5098: [BZOJ1098][POI2007]办公楼biu 没有数据结构就很棒 一个看上去非常玄学的代码 const int N=1e5+10,M=2e6+10; int n,m; int fa[ ...
- 类中嵌套定义指向自身的引用(C、C++、C#)或指针(C、C++)
在定义类的时候,类中可以嵌套定义指向自身的引用(C.C++.C#)或指针(C.C++).详见代码: Node类: using System; using System.Collections.Gene ...
- Spring Cloud-Eureka 服务注册中心
Eureka 是 Netflix 开发的,一个基于 REST 服务的,服务注册与发现的组件 它主要包括两个组件:Eureka Server 和 Eureka Client Eureka Client: ...
- GitLab的权限管理及Merge Request
GitLab的权限管理及Merge Request 原创尘世间一名迷途小码农 发布于2019-06-09 12:40:30 阅读数 2909 收藏 展开 目录 1.前言 2.角色权限 3.强制代码审 ...