C++ 智能指针的模拟实现实例

 更新时间:2017年07月26日 09:40:41   作者:Sweet_wen  
这篇文章主要介绍了C++ 智能指针的模拟实现实例的相关资料,智能指针是一个类,它把普通指针封装起来,能实现和普通指针同样的功能。,需要的朋友可以参考下

C++ 智能指针的模拟实现实例

1.引入

int main()
{
  int *p = new int;  //裸指针
  delete p;
  return 0;
}

在上面的代码中定义了一个裸指针p,需要我们手动释放。如果我们一不小心忘记释放这个指针或者在释放这个指针之前,发生一些异常,会造成严重的后果(内存泄露)。而智能指针也致力于解决这种问题,使程序员专注于指针的使用而把内存管理交给智能指针。

普通指针也容易出现指针悬挂问题,当有多个指针指向同一个对象的时候,如果某一个指针delete了这个对象,所以这个指针不会对这个对象进行操作,那么其他指向这个对象的指针呢?还在等待已经被删除的基础对象并随时准备对它进行操作。于是悬垂指针就形成了,程序崩溃也“指日可待”。

int main()
{
  int *p1 = new int(2);
  int *p2 = p1;
  int *p3 = p2;
  cout<<*p1<<endl;
  cout<<*p2<<endl;
  cout<<*p3<<endl;
  delete p1;
  cout<<*p2<<endl;
  return 0;
}

输出结果

2
2
2
-572662307

输出的结果*p2的结果并不是期待中2,因为2早已经被删除了。

智能指针

智能指针是一个类,它把普通指针封装起来,能实现和普通指针同样的功能。不同的是智能指针能够对内存进行自动管理,利用类对象出了作用域会调用析构函数,把对指针的释放写在析构函数中,避免出现悬挂指针的情况。

智能指针(smart pointer)是存储指向动态分配(堆)对象指针的类,用于生存期控制,能够确保自动正确的销毁动态分配的对象,防止内存泄露。它的一种通用实现技术是使用引用计数(reference count)。智能指针类将一个计数器与类指向的对象相关联,引用计数跟踪该类有多少个对象共享同一指针。每次创建类的新对象时,初始化指针并将引用计数置为1;当对象作为另一对象的副本而创建时,拷贝构造函数拷贝指针并增加与之相应的引用计数;对一个对象进行赋值时,赋值操作符减少左操作数所指对象的引用计数(如果引用计数为减至0,则删除对象),并增加右操作数所指对象的引用计数;调用析构函数时,构造函数减少引用计数(如果引用计数减至0,则删除基础对象)。

智能指针就是模拟指针动作的类。所有的智能指针都会重载 -> 和 * 操作符。智能指针还有许多其他功能,比较有用的是自动销毁。这主要是利用栈对象的有限作用域以及临时对象(有限作用域实现)析构函数释放内存。当然,智能指针还不止这些,还包括复制时可以修改源对象等。智能指针根据需求不同,设计也不同(写时复制,赋值即释放对象拥有权限、引用计数等,控制权转移等)。auto_ptr 即是一种常见的智能指针。

智能指针的实现(用类模板实现)

class Test
{
public:
  Test()
  {
    cout<<"Test()"<<endl;
  }
  ~Test()
  {
    cout<<"~Test()"<<endl;
  }
  void func()
  {
    cout<<"call Test::func()"<<endl;
  }
};
template<typename T>
class CSmartptr
{
public:
  CSmartptr(T *ptr):_ptr(ptr)
  {cout<<"CSmartptr()"<<endl;}
  CSmartptr(const CSmartptr<T> &other)
  {
    _ptr = new T;
    *ptr = *other._ptr;
  }
  ~CSmartptr()
  {
    cout<<"~CSmartptr()"<<endl;
    delete _ptr;
  }
  void relase() const
  {
    ((CSmartptr<T> *)this)->owns = false;
  }
  T& operator*()
  {
    return *_ptr;
  }
  const T& operator*()const {return *_ptr;}
  T *operator->()
  {
    return _ptr;
  }
  const T *operator->()const {return _ptr;}
private:
  T *_ptr;
};
int main()
{
  CSmartptr<int> p1(new int);
  *p1 = 200;
  CSmartptr<Test> p2(new Test);
  p2->func();
  return 0;
}

模拟实现auto_ptr

template<typename T>
class CSmartptr
{
public:
  CSmartptr(T *ptr):_ptr(ptr),owns(true){cout<<"CSmartptr()"<<endl;}
  CSmartptr(const CSmartptr<T> &other)
  {
    other.relase();
    _ptr = other._ptr;
  }
  ~CSmartptr()
  {
    cout<<"~CSmartptr()"<<endl;
    if( owns == true)
    {
      cout<<"~CSmartptr()"<<endl;
      delete _ptr;
    }

  }
  void relase() const
  {
    ((CSmartptr<T> *)this)->owns = false;
  }
  T& operator*()
  {
    return *_ptr;
  }
  const T& operator*()const {return *_ptr;}
  T *operator->()
  {
    return _ptr;
  }
  const T *operator->()const {return _ptr;}
private:
  T *_ptr;
  bool owns; //标志位 ,控制一个资源的访问权限
};
int main()
{
  CSmartptr<int> p1(new int);
  *p1 = 200;
  CSmartptr<Test> p2(new Test);
  p2->func();
  return 0;
}

带有引用计数的智能指针(方便对资源的管理和释放)

class CHeapTable
{
public:
  static CHeapTable& getInstance()
  {
    return mHeapTable;
  }
  //增加引用计数
  void addRef(void *ptr)
  {
    pthread_mutex_lock(mutex);
    list<Node>::iterator it = find(mList.begin(),
      mList.end(), ptr); // Node == Node it->mpaddr
    if(it == mList.end())
    {
      mList.push_front(ptr);
      cout<<"new addr:"<<ptr<<" ref:"<<1<<endl;
    }
    else
    {
      it->mcount++;
      cout<<"add addr:"<<ptr<<" ref:"<<it->mcount<<endl;
    }
    pthread_mutex_unlock(mutex);
  }
  //减少引用计数的
  void delRef(void *ptr)
  {
    list<Node>::iterator it = find(mList.begin(),
      mList.end(), ptr);
    if(it != mList.end())
    {
      it->mcount--;
      cout<<"del addr:"<<ptr<<" ref:"<<it->mcount<<endl;
      if(it->mcount == 0)
      {
        mList.erase(it);
      }
    }
  }
  //获取引用计数的
  int getRef(void *ptr)
  {
    list<Node>::iterator it = find(mList.begin(),
      mList.end(), ptr);
    if(it != mList.end())
    {
      return it->mcount;
    }
    return 0;
  }
private:
  CHeapTable(){}
  static CHeapTable mHeapTable;

  struct Node
  {
    Node(void *ptr=NULL):mpaddr(ptr),mcount(1){}
    bool operator==(const Node &src)
    {
      return mpaddr == src.mpaddr;
    }
    void *mpaddr; //标识堆内存资源
    int mcount; //标识资源的引用计数
  };

  list<Node> mList;
};
CHeapTable CHeapTable::mHeapTable;
template<typename T>
class CSmartPtr
{
public:
  CSmartPtr(T *ptr = NULL)
    :mptr(ptr)
  {
    if(mptr != NULL)
    {
      addRef();
    }
  }
  ~CSmartPtr()
  {
    delRef();
    if(0 == getRef())
    {
      delete mptr; 
      mptr = NULL;
    }
  }

  CSmartPtr(const CSmartPtr<T> &src)
    :mptr(src.mptr)
  {
    if(mptr != NULL)
    {
      addRef();
    }
  }

  CSmartPtr<T>& operator=(const CSmartPtr<T> &src)
  {
    if(this == &src)
      return *this;

    delRef();
    if(0 == getRef())
    {
      delete mptr;
      mptr = NULL;
    }

    mptr = src.mptr;
    if(mptr != NULL)
    {
      addRef();
    }
  }
  T& operator*(){return *mptr;}
  const T& operator*()const{return *mptr;}
  T* operator->(){return mptr;}
  const T* operator->()const{return mptr;}

  void addRef(){mHeapTable.addRef(mptr);}
  void delRef(){mHeapTable.delRef(mptr);}
  int getRef(){return mHeapTable.getRef(mptr);}
private:
  T *mptr;
   static CHeapTable &mHeapTable;
};
template<typename T>
CHeapTable& CSmartPtr<T>::mHeapTable = CHeapTable::getInstance();

以上就是智能指针的实例详解,如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持

相关文章

  • C语言趣味编程之平分七筐鱼

    C语言趣味编程之平分七筐鱼

    这篇文章介绍了C语言趣味编程之平分七筐鱼,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-11-11
  • C++使用LibCurl实现Web隐藏目录扫描功能

    C++使用LibCurl实现Web隐藏目录扫描功能

    LibCurl是一个开源的免费的多协议数据传输开源库,该框架具备跨平台性,开源免费,并提供了包括HTTP、FTP、SMTP、POP3等协议的功能,本文将给大家介绍C++使用LibCurl实现Web隐藏目录扫描功能
    2023-11-11
  • matlab GUI指纹识别门禁系统介绍及源码实现

    matlab GUI指纹识别门禁系统介绍及源码实现

    这篇文章主要为大家介绍了matlab GUI指纹识别门禁系统的介绍及源码实现,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
    2022-02-02
  • C语言简明分析指针与引用的具体用法

    C语言简明分析指针与引用的具体用法

    指针是一个实体,引用是一个别名;在汇编上,引用的底层是以指针的方式实现的,定义一个引用变量,相当于定义了一个指针,然后把引用内存的地址写到这个指针里面,当通过引用变量修改它所引用的内存时,它先访问了指针里面的地址,然后在这个地址的内存里面对值进行修改
    2022-05-05
  • 浅析c++ 宏 #val 在unicode下的使用

    浅析c++ 宏 #val 在unicode下的使用

    以下是对c++中宏#val在unicode下的使用方法进行了详细的分析介绍,需要的朋友可以参考下
    2013-07-07
  • C++ QT QThread启动、停止、暂停和恢复的实现

    C++ QT QThread启动、停止、暂停和恢复的实现

    本文主要介绍了C++ QT QThread启动、停止、暂停和恢复的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-06-06
  • C++实现扫雷程序开发

    C++实现扫雷程序开发

    这篇文章主要为大家详细介绍了C++实现扫雷程序开发,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • C语言指针引用数组案例讲解

    C语言指针引用数组案例讲解

    这篇文章主要介绍了C语言指针引用数组案例讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • C++ 类this及返回自身对象的引用方式

    C++ 类this及返回自身对象的引用方式

    这篇文章主要介绍了C++ 类this及返回自身对象的引用方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-11-11
  • Qt实现验证码相关功能的代码示例

    Qt实现验证码相关功能的代码示例

    验证码的原理基于人类视觉和计算机视觉的差异性,通过给用户显示一些难以被机器识别的图形或文字,让用户进行人机交互,确认自己的身份,这样可以有效保护网站安全,所以本给大家介绍了Qt实现验证码相关功能的代码示例,感兴趣的朋友可以参考下
    2024-01-01

最新评论