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分析的更多相关文章

  1. [4] 智能指针boost::scoped_ptr

    [1]boost::scoped_ptr简介 boost::scoped_ptr属于boost库,定义在namespace boost中,包含头文件#include <boost/scoped_ ...

  2. [5] 智能指针boost::shared_ptr

    [1]boost::shared_ptr简介 boost::shared_ptr属于boost库,定义在namespace boost中,包含头文件#include<boost/shared_p ...

  3. [6] 智能指针boost::weak_ptr

    [1]boost::weak_ptr简介 boost::weak_ptr属于boost库,定义在namespace boost中,包含头文件 #include<boost/weak_ptr.hp ...

  4. 详解 boost 库智能指针(scoped_ptr<T> 、shared_ptr<T> 、weak_ptr<T> 源码分析)

    一.boost 智能指针 智能指针是利用RAII(Resource Acquisition Is Initialization:资源获取即初始化)来管理资源.关于RAII的讨论可以参考前面的文章.在使 ...

  5. 【C/C++学院】0904-boost智能指针/boost多线程锁定/哈希库/正則表達式

    boost_array_bind_fun_ref Array.cpp #include<boost/array.hpp> #include <iostream> #includ ...

  6. 关于智能指针boost::shared_ptr

    boost库中的智能指针shared_ptr, 功能强大, 且开销小,故受到广大coder的欢迎. 但在实际的使用过程中,笔者也发现了一些不足. 1.定制的删除器 shared_ptr除了可以使用默认 ...

  7. C++ 智能指针 std::auto_ptr 分析

    背景介绍: RAll机制 定义一个类来封装资源的分配和释放,在构造函数中完成资源的分配和初始化,在析构函数中完成资源的清理,从而保证资源的正确初始化和清理 ps:智能指针就是RAll机制的一种应用,智 ...

  8. Qt智能指针简明说明

    下面的智能指针分别对应boost库,Qt库,c++11的智能指针 boost::scoped_ptr  QScopedPointer unique_ptr 在其生命期结束后会自动删除它所指的对象(确定 ...

  9. Boost中的智能指针(转)

    这篇文章主要介绍 boost中的智能指针的使用.(转自:http://www.cnblogs.com/sld666666/archive/2010/12/16/1908265.html) 内存管理是一 ...

随机推荐

  1. webapi HttpGet标签

    该标签可以指定路由如HttpGet["Test"],以前用的很顺,后来加了Area后,按照area/controller/Test的路径去访问报404,原因是HTTPGet指定路由 ...

  2. EFK收集nginx日志

    准备三台centos7的服务器 两核两G的 关闭防火墙和SELinux systemctl stop firewalld setenforce 0 1.每一台都安装jdk rpm -ivh jdk-8 ...

  3. cloudevents 通用event 描述指南

    cloudevents 是由cncf 组织管理的一个通用event描述指南 特性: 一致性 可理解性 可移植性 说明 cloudevents 不仅提供了核心描述,同时也包含了不同协议的指南说明(htt ...

  4. diffy 方便的bug 以及流量测试系统

    diffy 是twiiter 开源的流量以及bug 查找系统 参考使用图 几点说明 使用diffy我们需要三个角色 candidate instance 候选实例,运行新的代码 primary ins ...

  5. 洛谷 P1991 无线通讯网 题解

    P1991 无线通讯网 题目描述 国防部计划用无线网络连接若干个边防哨所.2 种不同的通讯技术用来搭建无线网络: 每个边防哨所都要配备无线电收发器:有一些哨所还可以增配卫星电话. 任意两个配备了一条卫 ...

  6. Linux下的零拷贝

    Reference:  https://segmentfault.com/a/1190000011989008 零拷贝是什么? 维基百科对“零拷贝”是这样描述的: "Zero-copy&qu ...

  7. OpenStack创建网络和虚拟机、dhcp设备、虚拟路由器、虚拟机访问外网原理分析

    创建网络和虚拟机流程: 1.创建网络和子网 背后发生了什么: Neutron让控制节点上针对此子网的dhcp(虚拟设备)启动,用于给该子网下的实例分配ip 2.生成虚拟机 背后发生了什么: 用户通过G ...

  8. HTTP协议之multipart/form-data

    HTTP协议之multipart/form-data REST API可以用multipart/form-data,来上传文件. 在最初的 http 协议中,没有上传文件方面的功能. rfc1867为 ...

  9. Dubbo的集群容错与负载均衡策略及自定义(一致性哈希路由的缺点及自定义)

    Dubbo的集群容错策略 正常情况下,当我们进行系统设计时候,不仅要考虑正常逻辑下代码该如何走,还要考虑异常情况下代码逻辑应该怎么走.当服务消费方调用服务提供方的服务出现错误时候,Dubbo提供了多种 ...

  10. Fragment: 使用newInstance()来实例化fragment(转)

    先说结论: 1. 当需要给Fragment传值时,使用newInstance()方式来实例化一个Fragment,能够更好的将该Fragment使用的参数捆绑起来,不必每次实例化时都写下面的代码: B ...