std::shared_ptr 是通过指针保持对象共享所有权的智能指针。多个 shared_ptr 对象可占有同一对象大概实现了一下,主要实现原理为,共享指针内部持有堆资源的指针以及引用计数的指针,通过对这两个指针的维护,达到多个共享对象对同一资源的控制

  实现主要分为三个文件。share_ptr.h,smart_ptr_define.h, main.cpp  (编译平台:Linux centos 7.0 编译器:gcc 4.8.5 )

  //smart_ptr_define.h
#ifndef __SMART_PTR_DEFINE_H__
#define __SMART_PTR_DEFINE_H__ #include <assert.h> #define PTR_ASSERT(x) assert(x) #define _SMART_PTR_BEGIN namespace smartptr {
#define _SMART_PTR_END }
#define _SMART_PTR ::smartptr:: #endif

  主要实现文件share_ptr.h

 #ifndef __SHARE_PTR_H__
#define __SHARE_PTR_H__ #include <iostream>
#include "smart_ptr_define.h" _SMART_PTR_BEGIN template <class T>
struct default_deleter
{
void operator()(T* ptr)
{
if (ptr != NULL)
{
delete ptr;
ptr = NULL;
}
}
}; template <class T, class deleter = default_deleter<T> >
class shared_ptr
{
public:
typedef shared_ptr<T, deleter> SHARE_PTR; shared_ptr()
{
m_ptr = NULL;
m_iRefCount = NULL;
} explicit shared_ptr(T* ptr)
{
if (ptr != NULL)
{
m_ptr = ptr;
RefCountInit();
}
} shared_ptr(deleter d, T* ptr)
{
if (ptr != NULL)
{
m_ptr = ptr;
m_deleter = d;
RefCountInit();
}
} //拷贝构造
shared_ptr(const SHARE_PTR& sh_ptr)
{
if (sh_ptr.m_ptr != NULL)
{
m_ptr = sh_ptr.m_ptr;
m_deleter = sh_ptr.m_deleter;
m_iRefCount = sh_ptr.m_iRefCount; RefCountIncrease();
}
} //赋值运算符
SHARE_PTR& operator = (const SHARE_PTR& sh_ptr)
{
if (this != &sh_ptr)
{
RefCountDecrease(); if (sh_ptr.m_ptr != NULL)
{
m_ptr = sh_ptr.m_ptr;
m_deleter = sh_ptr.m_deleter;
m_iRefCount = sh_ptr.m_iRefCount; RefCountIncrease();
}
} return (*this);
} ~shared_ptr()
{
RefCountDecrease();
} public:
//提领操作
T& operator*()
{
PTR_ASSERT(m_ptr != NULL);
return *(m_ptr);
} //原始指针操作
T* operator->()
{
PTR_ASSERT(m_ptr != NULL);
return m_ptr;
} operator bool() const
{
return m_ptr != NULL;
} //取得原始指针
T* getPointer()
{
PTR_ASSERT(m_ptr != NULL);
return m_ptr;
} //获得引用计数
int getRefCount()
{
PTR_ASSERT(m_iRefCount != NULL);
return *m_iRefCount;
} private:
void RefCountInit()
{
m_iRefCount = new int();
} void RefCountIncrease()
{
if (m_iRefCount != NULL)
{
++(*m_iRefCount);
}
} void RefCountDecrease()
{
if (m_iRefCount != NULL && --(*m_iRefCount) == )
{
m_deleter(m_ptr);
delete m_iRefCount;
m_ptr = NULL;
m_iRefCount = NULL;
}
} private:
int* m_iRefCount; //引用计数 T* m_ptr; //对象指针 deleter m_deleter; //删除器
}; _SMART_PTR_END
#endif // !__SHARE_PTR_H__

  main函数测试

 #include "share_ptr.h"
#include <memory> class Test
{
public:
Test()
{
std::cout << "construct.." << std::endl;
} void method()
{
std::cout << "welcome Test.." << std::endl;
} ~Test()
{
std::cout << "destruct.." << std::endl;
}
}; int main()
{
Test* t1 = new Test(); _SMART_PTR shared_ptr<Test> shptr(t1); _SMART_PTR shared_ptr<Test> shptr1(shptr); _SMART_PTR shared_ptr<Test> shptr2 = shptr1; std::cout << "RefCount: " << shptr2.getRefCount() << std::endl; shptr2->method(); (*shptr2).method(); if (shptr2)
{
std::cout << "ptr is exit " << std::endl;
} return ;
}

  测试最后打印:

 [yejy@yejy cmake-]$ ./smartptr
construct..
RefCount:
welcome Test..
welcome Test..
ptr is exit
destruct..
[yejy@yejy cmake-]$

  shared_ptr主要需实现的功能点如下(以下总结引用自网络,非原创):

  1. 没有参数构造的时候,初始化为空,即对象和引用计数的两个指针都为0

  2. 使用指针为参数构造时,拥有此指针,在没有智能指针指向它时进行析构

  3. 智能指针复制时,两个智能指针共同拥有内部指针,引用计数同时+1

  4. 智能指针可以使用智能指针或普通指针重新赋值。重载=操作符,对于智能指针赋值,需要考虑是否自赋值,以避免将自身析构了后再重新赋值,而普通指针赋值给智能指针,则不需要考虑自赋值,因为两者本身是两个类型

  5. 获得底层指针的访问,定义getPtrPointer()getPtrCounter()来分别返回底层指针和引用计数,定义operator bool()来处理智能指针隐式转换为bool的情况

  6. 重载->×操作符 ,来实现与普通指针相同的指针访问

  7. 需要支持隐式指针类型转换,static_cast不支持而dynamic_cast支持的转换则使用Cast<T2>()成员函数来解决。考虑定义友元类,以防止指向派生类的智能指针有权限访问基类的内部对象;当转型不成功时,返回为空 (未实现)

  8. 如果一个裸指针直接用来创建两个智能指针的话,期望的情况是当两个智能指针析构掉的时候,该指针会被delete两次从而崩溃(这是shared_ptr的特点)

  9. 不处理循环引用(也是shared_ptr的特点),可以通过与weak_ptr协作来打破循环

  10. 实现deleter机制

智能指针之 shared_ptr的更多相关文章

  1. 智能指针之shared_ptr基本概述

    1.shared_ptr允许有多个指针指向同一个对象,unique_ptr独占所指向的对象. 2.类似于vector,智能指针也是模板.创建智能指针: shared_ptr<string> ...

  2. 【C++11新特性】 C++11智能指针之shared_ptr

    C++中的智能指针首先出现在“准”标准库boost中.随着使用的人越来越多,为了让开发人员更方便.更安全的使用动态内存,C++11也引入了智能指针来管理动态对象.在新标准中,主要提供了shared_p ...

  3. C++智能指针之shared_ptr与右值引用(详细)

    1. 介绍 在 C++ 中没有垃圾回收机制,必须自己释放分配的内存,否则就会造成内存泄露.解决这个问题最有效的方法是使用智能指针(smart pointer).智能指针是存储指向动态分配(堆)对象指针 ...

  4. C++ | 再探智能指针(shared_ptr 与 weak_ptr)

    上篇博客我们模拟实现了 auto_ptr 智能指针,可我们说 auto_ptr 是一种有缺陷的智能指针,并且在C++11中就已经被摈弃掉了.那么本章我们就来探索 boost库和C++11中的智能指针以 ...

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

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

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

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

  7. 智能指针(二):shared_ptr实现原理

    前面讲到auto_ptr有个很大的缺陷就是所有权的转移,就是一个对象的内存块只能被一个智能指针对象所拥有.但我们有些时候希望共用那个内存块.于是C++ 11标准中有了shared_ptr这样的智能指针 ...

  8. 【STL学习】智能指针之shared_ptr

    前面已经学习过auto_ptr,这里补充另外一种智能指针,比auto_ptr要更强力更通用的shared_ptr. shared_ptr 简介及使用选择  几乎所有的程序都需要某种形式的引用计数智能指 ...

  9. 智能指针auto_ptr & shared_ptr

    转载:智能指针auto_ptr 很多人听说过标准auto_ptr智能指针机制,但并不是每个人都天天使用它.这真是个遗憾,因为auto_ptr优雅地解决了C++设计和编码中常见的问题,正确地使用它可以生 ...

随机推荐

  1. 【Unity Shaders】Diffuse Shading——使用2D ramp texture来创建一个假的BRDF(双向反射分布函数)

    本系列主要参考<Unity Shaders and Effects Cookbook>一书(感谢原书作者),同时会加上一点个人理解或拓展. 这里是本书所有的插图.这里是本书所需的代码和资源 ...

  2. AngularJS进阶(二十八)解决AngualrJS页面刷新导致异常显示问题

    解决AngualrJS页面刷新导致异常显示问题 绪 俗话说,细节决定成败,编程亦是如此.编程过程中我们可能会不自觉的忽视一些细节问题,殊不知,这些细节正是导致页面显示出现问题的地方.今略举一例,与君共 ...

  3. 学习OpenCV,GPU模块

    如何使用opencv的gpu库呢?我这两天一直在搞这个事情,环境的配置见上文(转载),这里我先举个简单的例子,实现这样的功能:host读入一幅图像,加载到GPU上,在GPU上复制一份然后下传到host ...

  4. Android系统修改硬件设备访问权限

    Android系统修改硬件设备访问权限 在硬件抽象层模块文件(so)文件中,提供的函数调用open函数来打开设备文件,比如/dev/gpio,如果不修改设备文件/dev/gpio的访问权限,那么应用程 ...

  5. 网站论坛同步用户,整合api,实现…

    在网上参考了很多资料后,终于完美实现了网站和discuz!nt论坛的双向整合,整合后网站和论坛之间可以同步注册.登录.退出和修改登录密码操作. 本系统的实现形式是新云CMS网站(ASP)和Discuz ...

  6. OV5640全景模式预览倒180度,拍照正常的问题

    此方法基本上适用于所有android平台上全景模式预览倒180度,拍照正常的问题. 首先说明的是,影响camera方向的有三个地方,分别是系统方向,内核camera方向和驱动镜像.全景模式预览只跟系统 ...

  7. Material Design5.x动画实现解析篇一

    Material Design设计语言动画篇共推出六种类型的动画效果: 1.Touch feedback(触摸反馈) 2.Reveal effect(揭露效果) 3.Activity transiti ...

  8. The type java.lang.Object cannot be resolved. It is indirectly referenced from required .class files

    The type java.lang.Object cannot be resolved.It is indirectly referenced from required .class files ...

  9. SharePoint 2010 之soap:Server服务器无法处理请求

    算是一个下马威?!刚刚部署上的SharePoint2010环境,感觉很欣喜,开始操作,结果装上Designer2010,打开页面,居然报错 错误内容:soap:Server服务器无法处理请求. --- ...

  10. GEFGWT的HelloWorld

    发现一个好玩的东西,gef-gwt,使用它可以轻松的在web上建立gef程序,原文在这里http://gefgwt.org/getting-started/(文章虽然是英文,但是很容易懂,我是按部就班 ...