博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
引用计数实现
阅读量:6375 次
发布时间:2019-06-23

本文共 5540 字,大约阅读时间需要 18 分钟。

hot3.png

在看More Effective C++时就有引用计数,现在自己实现了下。

完成计数的基类

/**************************************************author:周翔*e-mail:604487178@qq.com*blog:http://blog.csdn.net/zhx6044***************************************************/#ifndef REFCOUNTER_H#define REFCOUNTER_H#include 
class RefCounter{public: RefCounter(); virtual ~RefCounter() = 0; void plusOneRef(); void minusOneRef(); bool isShared() const;private: int m_refCounter;};#endif // REFCOUNTER_H/**************************************************author:周翔*e-mail:604487178@qq.com*blog:http://blog.csdn.net/zhx6044***************************************************/#include "refcounter.h"RefCounter::RefCounter():m_refCounter(1){}RefCounter::~RefCounter(){}void RefCounter::plusOneRef(){ ++m_refCounter;}void RefCounter::minusOneRef(){ --m_refCounter;}bool RefCounter::isShared() const{ return m_refCounter > 1;}
一个需要实现引用计数的类

/**************************************************author:周翔*e-mail:604487178@qq.com*blog:http://blog.csdn.net/zhx6044***************************************************/#ifndef OBJECT_H#define OBJECT_H#include 
#include "refcounter.h"class Object : public RefCounter{ friend Object operator +(const Object& _v1, const Object& _v2);public: Object(const char* _value); virtual ~Object(); Object(const Object& _v); const char* className() const;private: char *pd;};#endif // OBJECT_H/**************************************************author:周翔*e-mail:604487178@qq.com*blog:http://blog.csdn.net/zhx6044***************************************************/#include "object.h"#include
Object::Object(const char *_value){ pd = new char[strlen(_value) + 1]; strcpy(pd, _value);}Object::~Object(){ if (!isShared()) { delete[] pd; }}Object operator +(const Object& _v1, const Object& _v2){ return Object((std::string(_v1.pd) + _v2.pd).data());}Object::Object(const Object &_v){ pd = new char[strlen(_v.pd) + 1]; strcpy(pd,_v.pd);}const char* Object::className() const{ return pd;}

一个实现引用判断的外裹类模板,封装了逻辑

/**************************************************author:周翔*e-mail:604487178@qq.com*blog:http://blog.csdn.net/zhx6044***************************************************/#ifndef REFCOUNTERWAPPER_H#define REFCOUNTERWAPPER_H#include 
template
class RefCounterWapper{public: RefCounterWapper(T *p); ~RefCounterWapper(); RefCounterWapper(const T& _v); RefCounterWapper(const RefCounterWapper &_v); RefCounterWapper& operator =(const RefCounterWapper& _v); const T* operator ->() const; const T& operator *() const; T* operator ->(); T& operator *();private: T *pd;};template
RefCounterWapper
::RefCounterWapper(T *p):pd(p){}template
RefCounterWapper
::~RefCounterWapper(){ if (!pd->isShared()) { delete pd; } else { pd->minusOneRef(); }}template
RefCounterWapper
::RefCounterWapper(const T &_v){ pd = new T(_v);}template
RefCounterWapper
::RefCounterWapper(const RefCounterWapper &_v){ pd = _v.pd; pd->plusOneRef();}template
RefCounterWapper
& RefCounterWapper
::operator =(const RefCounterWapper& _v){ if (this == &_v) return *this; pd->minusOneRef(); if (!pd->isShared()) { delete pd; } pd = _v.pd; pd->plusOneRef(); return *this;}template
const T* RefCounterWapper
::operator ->() const{ return pd;}template
const T& RefCounterWapper
::operator *() const{ return *pd;}template
T* RefCounterWapper
::operator ->(){ return pd;}template
T& RefCounterWapper
::operator *(){ return *pd;}#endif // REFCOUNTERWAPPER_H

供外部使用的具有计数功能的类

/**************************************************author:周翔*e-mail:604487178@qq.com*blog:http://blog.csdn.net/zhx6044***************************************************/#ifndef REFCOUNTEROBJECT_H#define REFCOUNTEROBJECT_H#include "refcounterwapper.h"#include "object.h"class RefCounterObject{    friend RefCounterObject operator +(const RefCounterObject& _v1, const RefCounterObject& _v2);public:    RefCounterObject(const char* _v = "Object");    virtual ~RefCounterObject();    const char* className() const;private:    RefCounterObject(const Object& _v);    RefCounterWapper pd;};RefCounterObject::RefCounterObject(const char *_v):pd(new Object(_v)){}RefCounterObject::~RefCounterObject(){}RefCounterObject operator +(const RefCounterObject& _v1, const RefCounterObject& _v2){    return RefCounterObject((*_v1.pd) + (*_v2.pd));}RefCounterObject::RefCounterObject(const Object &_v):pd(_v){}const char* RefCounterObject::className() const{    return pd->className();}#endif // REFCOUNTEROBJECT_H
测试代码

#include 
#include "refcounterobject.h"using namespace std;int main(){ RefCounterObject v("zhou love cc"); cout << v.className() << endl; RefCounterObject v2 = v; cout << v2.className() << endl; RefCounterObject v3(v2); cout << v3.className() << endl; RefCounterObject v4; cout << v4.className() << endl; v4 = v3; cout << v4.className() << endl; RefCounterObject v5 = v4 + v2; cout << v5.className() << endl;}

5个对象共享了2个Object,它们保存的字符串的值,分别位zhou love cc和zhou love cczhou love cc。

用Valgrind分析了内存,没有内存泄漏

分析Object的析构函数被调用几次

5个对象有两个Object对象,调用析构2次,

还有各一次是

RefCounterObject v4;    cout << v4.className() << endl;    v4 = v3;//析构v4开始的Object    cout << v4.className() << endl;    RefCounterObject v5 = v4 + v2;//+操作底层产生一个局部的Object    cout << v5.className() << endl;

共享的是RefCounterWapper中pd这个指针指向的对象。

这几个类的关系如图

转载于:https://my.oschina.net/u/854744/blog/418367

你可能感兴趣的文章
HDU 3068 最长回文(manacher算法)
查看>>
二叉树
查看>>
Node脚手架编写初学者教程
查看>>
08_Node js 工具模块 util
查看>>
手把手教你如何安装水晶易表——靠谱的安装教程
查看>>
Python单例模式(Singleton)的N种实现
查看>>
requirejs的插件介绍与制作
查看>>
SpringBoot整合Angular应用第二弹-配置支持Angular
查看>>
Facebook、纽约大学利用机器学习5分钟搞定核磁共振检查
查看>>
221. Maximal Square
查看>>
MySQL基础
查看>>
机器学习A-Z~支持向量机
查看>>
PAT A1010 二分进制结合重点题
查看>>
LeetCode35.搜索插入位置 JavaScript
查看>>
5个让人赞不绝口的微信小程序,拒绝占用手机内存!
查看>>
Spring Security整合KeyCloak保护Rest API
查看>>
POS概述
查看>>
containerd发布了CRI修复程序和CVE-2019-5736更新的runc
查看>>
77. Combinations
查看>>
WEB前端开发的思考与感悟
查看>>