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. 我的Json解析实战

    所谓json,其实就是在我们访问一个网页的接口的时候,服务器端传送给我们客户端的一种数据的结构,当然我们向服务器端发送的数据有时也会转换成json格式,当然了,这不是必须的.最近在解析一些json字符 ...

  2. 基于FT5x06嵌入式Linux电容触摸屏驱动

    **************************************************************************************************** ...

  3. 高通 8x12 添加 TP和按键

    1 .在tp的驱动文件中添加以下代码实现按键功能 [plain] view plain copy static ssize_t ft5x06_virtual_keys_register(struct  ...

  4. Linux管道编程实例

    /*管道 可以把管道想象为两个实体之间的单向连接器.注意,管道是半双工的, 如果需要全双工通讯,应该转而考虑套接字. 匿名管道又称管道,提供了一个进程与它的兄弟进程通讯的方法,只存在于父进程中: 命名 ...

  5. Locally Weighted Linear Regression 局部加权线性回归-R实现

      局部加权线性回归  [转载时请注明来源]:http://www.cnblogs.com/runner-ljt/ Ljt 作为一个初学者,水平有限,欢迎交流指正. 线性回归容易出现过拟合或欠拟合的问 ...

  6. jsp中的tag与tld

    转载自: http://www.cnblogs.com/fanzi2009/archive/2010/04/08/1707888.html 在jsp文件中,可以引用tag和tld文件.  1.对于ta ...

  7. 【面试笔试算法】Problem 9: 腾讯2016年研发实习笔试题:最长回文子串

    (一)题目 问题:求给定字符串s的回文(palindrome)子串中,长度最大的回文子串的长度. 回文(palindrome)是指从左往右读和从右往左读字符串,看到的字符串都是一样的.比如" ...

  8. Gradle 1.12用户指南翻译——第二十四章. Groovy 插件

    其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...

  9. cocos2d-x 游戏开发之有限状态机(FSM) (三)

    cocos2d-x 游戏开发之有限状态机(FSM) (三) 有限状态机简称FSM,现在我们创建一个专门的FSM类,负责管理对象(Monkey)的状态.然后Monkey类就实现了行为与状态分离.Monk ...

  10. Android性能优化典例(一)

    在Android开发过程中,很多时候往往因为代码的不规范.api使用不恰当.控件的使用场景考虑不全面和用户不恰当的操作等都能引发一系列性能问题的,下面就是我目前整理的一些Android开发过程中需要注 ...