智能指针shared_ptr【无锁设计基于GCC】

1. shared_ptr 介绍

  使用过Boost的话对shared_ptr一定有很深的印象。多个shared_ptr指向同一个对象,每个shared_ptr会使对象的引用计数加+1,当引用计数为0时,

对象将被析构。本文实现一个简洁版本的shared_ptr,并没有太多跨平台特性,实现代码可以再GCC上运行。

      本文中的引用计数由ref_count_t类实现,参见下文的详细分析。

  详文另见:

  代码详见:http://ffown.sinaapp.com/?p=49

  svn co http://ffown.googlecode.com/svn/trunk/fflib/lib

2.  shared_ptr 的构造

     我们期望shared_ptr的行为尽量的接近原始指针的行为。所以shared_ptr应该支持三种构造方式

  a. 空构造类似与void* p =NULL;

      b. shared_ptr可以通过原始对象指针构造,类似于void* p = q;

  c. shared_ptr 可以通过已存在的shared_ptr构造。

  首先shared_ptr是一个模板类,其由连个属性。

private:
    object_t*       m_dest_ptr;
    ref_count_t*    m_ref_count;
};

其中m_dest_ptr 指向目标对象, m_ref_count 用来记录该对象的引用计数。为了简单,shared_ptr类遵循一个原则m_dest_ptr和m_ref_count  同时为NULL,或同时不为NULL。

其中 object_t 为模板类型的别名。

template<typename T>
class shared_ptr_t
{
public:
    typedef T               object_t;
    typedef shared_ptr_t<T> self_type_t;

  1> 空构造目标对象和引用计数默认都为空。

template<typename T>
shared_ptr_t<T>::shared_ptr_t(object_t* p):
    m_dest_ptr(p),
    m_ref_count(NULL)
{
    if (NULL != m_dest_ptr)
    {
        m_ref_count = new ref_count_t();
    }
}

  share_ptr_t<int> p;

  2> 支持原始对象指针作为构造函数参数

template<typename T>
shared_ptr_t<T>::shared_ptr_t(object_t* p):
    m_dest_ptr(p),
    m_ref_count(NULL)
{
    if (NULL != m_dest_ptr)
    {
        m_ref_count = new ref_count_t();
    }
}

  用例:share_ptr_t<int> p(new int());   3> 使用已存在的shared_ptr 构造

template<typename T>
shared_ptr_t<T>::shared_ptr_t(self_type_t& p):
    m_dest_ptr(p.get()),
    m_ref_count(p.ger_ref_count())
{
    if (NULL != m_dest_ptr)
    {
        m_ref_count->inc();
    }
}

  用例: share_ptr_t<int> q(p);

3. shared_ptr 获取引用计数或原始指针

  有时需要知道shared_ptr当前引用计数的值,通过shared_ptr获取原始指针理所当然。So:

size_t       ref_count() const       { return m_ref_count != NULL? (size_t)m_ref_count->size(): 0; }

所以很容易验证shared_ptr的行为:

shared_ptr_t p(new int());
assert(p.ref_count() == 1);
shared_ptr_t<int> q(p);
assert(q.ref_count() == 1);

4. 减少引用计数

  shared_ptr需要显示的析构对象,所以提供reset接口,当目标对象已经创建并且引用计数达到零时(即不再有shared_ptr保存目标对象的控制权),析构目标对象。

template<typename T>
void shared_ptr_t<T>::reset()
{
    if (m_dest_ptr)
    {
        if (true == m_ref_count->dec_and_check_zero())
        {
            delete m_ref_count;
            delete m_dest_ptr;
        }
        m_ref_count = NULL;
        m_dest_ptr = NULL;
    }
}

5. shared_ptr 的析构

  很简单,减少引用计数。

template<typename T>
shared_ptr_t<T>::~shared_ptr_t()
{
    reset();
}

6. 向原始指针一样使用shared_ptr

  可以这样使用shared_ptr

  struct foo_t { int a; }

  shared_ptr_t<foo_t> p(new foo_t());

  (*p).a = 100;

  p->a = 100;

  if(p) cout << "p not null!\n";

     所以提供如下接口:

template<typename T>
typename shared_ptr_t<T>::object_t&    shared_ptr_t<T>::operator*()
{
    assert(NULL != m_dest_ptr);
    return *m_dest_ptr;
}

template<typename T>
typename shared_ptr_t<T>::object_t*    shared_ptr_t<T>::operator->()
{
    assert(NULL != m_dest_ptr);
    return m_dest_ptr;
}

7. Lock Free引用计数实现

  GCC中已经定义了一些atomic operation,但是查阅资料后,应该是对Intel的平台支持较好,其他平台支持不确定。故把atomic操作封装成宏。

#define ATOMIC_ADD(src_ptr, v)                         (void)__sync_add_and_fetch(src_ptr, v)
#define ATOMIC_SUB_AND_FETCH(src_ptr, v)  __sync_sub_and_fetch(src_ptr, v)

  ref_count_t 实现很简单:

class ref_count_t
{
    typedef  volatile long atomic_t;
public:
    ref_count_t():
        m_ref_num(1)
    {}
    ~ref_count_t()
    {}

    inline void inc()
    {
        ATOMIC_ADD(&m_ref_num, 1);
    }
    inline bool dec_and_check_zero()
    {
        return 0 == ATOMIC_SUB_AND_FETCH(&m_ref_num, 1);
    }
    inline atomic_t size()
    {
        return m_ref_num;
    }

private:
    atomic_t m_ref_num;
};
#endif

8. 线程安全性

1. 单线程多个shared_ptr指向不同的对象,安全。

2. 单线程多个shared_ptr指向相同的对象,安全。

3. 多线程多个操作不同的shared_ptr, 指向不同的对象,安全。

4. 多线程多个操作不同的shared_ptr, 指向相同对象,shared_ptr安全(也就是引用计数维护正确),对于原始对象操作依赖于用户。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏mathor

KMP(2)

994
来自专栏Charlie's Road

类方法load和initialize的区别

Objective-C作为一门面向对象语言,有类和对象的概念。编译后,类相关的数据结构会保留在目标文件中,在运行时得到解析和使用。在应用程序运行起来的时候,类的...

774
来自专栏王小雷

Python之数据规整化:清理、转换、合并、重塑

Python之数据规整化:清理、转换、合并、重塑 1. 合并数据集 pandas.merge可根据一个或者多个不同DataFrame中的行连接起来。 panda...

2036
来自专栏Albert陈凯

Scala Collections集合的几个重要概念

几个重要的概念 谓词是什么(What a predicate is) A predicate is simply a method, function, or...

2223
来自专栏cloudskyme

一步一步学lucene——(第四步:搜索篇)

下面说的主要是lucene如何进行搜索,相比于建索引,搜索可能更能提起大家的兴趣。 lucene的主要搜索的API 下面通过表格来看一下lucene用到的主要的...

3876
来自专栏好好学java的技术栈

Java提升篇:对象克隆(复制)

1133
来自专栏我是攻城师

Apache Pig学习笔记(二)

3369
来自专栏PHP技术

新手快速学习ES6语法,用最快的速度入门ES6就看这里

最近正在学习ES6,对于ES6的语法有一些自己的理解,想写这篇文章帮助跟我一样的新手快速入门ES6而不至于连代码都看不懂.至于开发环境的搭建什么的例如balel...

1073
来自专栏一个会写诗的程序员的博客

第9章 文件IO操作、正则表达式与多线程第9章 文件IO操作、正则表达式与多线程

我们在《第6章 扩展函数与属性》中已经介绍过Kotlin中的类扩展的特性。使用Kotlin的扩展函数功能,我们可以直接为 String 类实现一个 inc() ...

793
来自专栏PHP技术

PHP字符串和数组操作

*字符串查找 $email = 'name@example@.com'; $domain = strstr($email, '@'); echo $do...

3104

扫码关注云+社区