美文网首页
内存垃圾管理(智能指针)

内存垃圾管理(智能指针)

作者: 小幸运Q | 来源:发表于2021-06-01 21:14 被阅读0次

智能指针 shared_ptr:
https://blog.csdn.net/u012501459/article/details/48229399

C++11中引入了智能指针的概念。使用普通指针,容易造成堆内存泄露(忘记释放),二次释放,使用智能指针能更好的管理堆内存。

  • 构造函数中创建类的新对象时,初始化引用计数为1;
  • 拷贝构造函数复制指针,并使相应的引用计数增加1;
  • 赋值操作减少左操作数所值对象的引用计数,增加右操作数所指对象的引用计数;
  • 析构函数使引用计数减少1,并且当引用计数为0时,释放指针说指向的对象;


    image.png

定制删除器:

#include<iostream>
#include<memory>
using namespace std;
int main(){
    shared_ptr<int> sp1(new int(10));
    cout<<*sp1<<endl;
    *sp1 = 20;
    cout<<*sp1<<endl;
        return 0;
}

我们动态申请的内存,构造时完成初始化,析构时通过delete来释放没有任何问题。但是文件指针可能就挂了。

#include<iostream>
#include<memory>
using namespace std;
int main(){
    shared_ptr<FILE> sp1(fopen("test.txt","w"));
    //如果不进行任何特殊处理,则程序崩溃!
    return 0;
}

因此,shared_ptr提供了删除器功能,我们可以通过定制删除器,来对特定的资源进行回收。

#include<iostream>
#include<memory>
using namespace std;
 
//仿函数
struct Fclose{
    void operator()(void *ptr){
        fclose((FILE*)ptr);
        cout<<"不用担心,shared_ptr已经通过调用我关闭了文件!"<<endl;
    }
};
 
int main(){
    //我们通过传入仿函数,来完成对文件指针的清理
    shared_ptr<FILE> sp1(fopen("test.txt","w"),Fclose());
    return 0;
}

Ref_ptr类:

//使用int*指针初始化ptr,注意必须要放在初始化列表中
Ref_ptr(int * i):ptr(new Referenced(i))
{

}
//拷贝构造函数,又有一个变量指向了这块内存
Ref_ptr(const Ref_ptr & rhs)
{
    ptr=rhs.ptr;//将右操作数的引用计数对象赋值给左操作数
    ptr->ref();//将它们的应用计数加1
}

Ref_ptr r1=new int(4); //调用构造函数
Ref_ptr r2=r1; //调用拷贝构造函数
image.png
//赋值操作符,右操作数的引用计数要减1,左操作数的引用计数要加1
Ref_ptr & operator=(const Ref_ptr & rhs)
{
    if(&rhs==this)
        return *this;
    if(ptr->unref()==0)//赋值操作符,首先将当前类的引用计数减1,因为现在指向它的指针少了一个。
    {
        cout<<"delete Ref_ptr"<<endl;
        delete ptr;
    }
    ptr=rhs.ptr; //将右操作数的引用计数赋值给当前对象
    ptr->ref(); //引用计数加1
    return *this;
}

//析构函数,引用计数要减1,如果减为0,删除这块内存
~Ref_ptr()
{
    if(ptr->unref()==0)
    {
        cout<<"delete Ref_ptr"<<endl;
        delete ptr;
    }
}

Referenced类:

//初始化这个类,引用计数设为1,并且将p指向传入的地址
Referenced(int * pi)
{
    refCount=1;
    p=pi;
}

//引用计数加1
int ref()
{
    return ++refCount;
}   

//引用计数减1
int unref()
{
    return --refCount;
}

//析构函数,释放掉内存
~Referenced()
{
    cout<<"delete referenced"<<endl;
    delete p;
}

好处:

  • 智能指针能够帮助我们处理资源泄露问题;
  • 它也能够帮我们处理空悬指针的问题;
  • 它还能够帮我们处理比较隐晦的由异常造成的资源泄露。

循环引用:

image.png image.png
#include <memory>
class B;
class A
{
public:
    ~A()
    {
        printf("delete A");
    }
    std::shared_ptr < B >
        m_b;
};
 
class B
{
public:
    ~B()
    {
        printf("delete B");
    }
    std::shared_ptr < A >
        m_a;
};
 
int main()
{
    std::shared_ptr<A> a(new A); //new出来的A的引用计数此时为1
    std::shared_ptr<B> b(new B); //new出来的B的引用计数此时为1
    a->m_b = b; //B的引用计数增加为2
    b->m_a = a; //A的引用计数增加为2
}

相关文章

  • 智能指针和垃圾回收

    堆内存管理:智能指针与垃圾回收 显式内存管理 野指针 重复释放 内存泄漏 C++11 的智能指针 unique_p...

  • 内存垃圾管理(智能指针)

    智能指针 shared_ptr:https://blog.csdn.net/u012501459/article/...

  • C++ 智能指针

    智能指针 传统指针存在的问题需要手动管理内存容易发生内存泄露(忘记释放,出现异常等)释放之后产生野指针 智能指针就...

  • C++ 11 常用特性的使用经验总结(二)

    4、智能指针内存管理 在内存管理方面,C++11的std::auto_ptr基础上,移植了boost库中的智能指针...

  • Rust智能指针

    智能指针 在C/C++中,堆内存的申请和释放都由程序员自己管理,自C++11起,引入了智能指针来协助管理内存。对于...

  • 第十六章 string类和标准模板库(2)智能指针模板类

    (二)智能指针模板类 智能指针是行为类似指针的类对象,但这种对象还有其他便于管理内存的功能。 1.使用智能指针 (...

  • C++11——动态内存

    智能指针 为了使管理动态内存更容易、更安全,新标准库提供了两种管理动态对象的智能指针类型。智能指针的作用类似于常规...

  • 智能指针学习

    智能指针 介绍 为了更容易(同时也更安全的管)的使用动态内存,C++11提供了智能指针来管理new出来的内存sha...

  • C++智能指针

    智能指针其实本质是一个模板类,一般使用是用的这个类的对象,而不是指针智能指针体现在内存释放问题,用智能指针管理ne...

  • 源码分析shared_ptr实现

    智能指针是C++中一项很常用的技术,合理的使用智能指针可以更方便的管理内存,降低内存泄漏的风险,这里只介绍C++1...

网友评论

      本文标题:内存垃圾管理(智能指针)

      本文链接:https://www.haomeiwen.com/subject/jylltftx.html