C++11之智能指针(unique_ptr、shared_ptr、weak_ptr、auto_ptr)浅谈内存管理

简介: C++11之智能指针(unique_ptr、shared_ptr、weak_ptr、auto_ptr)浅谈内存管理

前言

下面这段代码看起来正常,但事实在特殊情况下f函数可能无法释放这个a资源。

void f()
{
  A * a = new A();
  ...
  delete a;
}

例如:


1.如果在中间这段代码中有一个过早的return语句,且刚好被执行那么就会出现内存泄漏,这时你可能会说在每个return前都加一个delete a;语句。不得不说这种做法可以避免上述出现的问题,但会导致代码中随处可见的delete a;语句显得很乱且臃肿。

2.如果在中间这段代码中出现一个异常,也会导致内存泄漏。

单纯靠着delete a;语句去释放内存是行不通的。

智能指针

这时就要用到智能指针这个东西了。自动释放内存

  1. 智能指针会帮程序员来管理对象。
  2. 在智能智能释放时自动会调用所管理对象的析构函数。


头文件:memory

使用方法

虽然智能指针种类较多但是用法都一样,下面以unique_ptr为例。在申请到内存时将会交给unique_ptr接管,在执行完f函数后a对象将会自动调用析构函数。

void f()
{
    ...
  std::unique_ptr<A> a(new  A());
  ...
}


unique_ptr

对象的所有权可以从一个独占指针转移到另一个指针,

转移方式:对象始终只能有一个指针作为其所有者。

当独占指针离开其作用域或将要拥有不同的对象时,它会自动释放自己所管理的对象。


实现unique_ptr类

template<class T>
class Unique_ptr
{
public:
    Unique_ptr(T* ptr)
    {
        this->ptr = ptr;
    }
    ~Unique_ptr()
    {
        if (ptr != nullptr)
        {
            delete ptr;
            cout << "释放ptr对象" << endl;
            ptr = nullptr;
        }
    }
    T& operator*()
    {
        return *ptr;
    }
    T* get()
    {
        return ptr;
    }
    T* operator->()
    {
        return ptr;
    }
private:
    Unique_ptr(const Unique_ptr& other) = delete;
    Unique_ptr& operator=(const Unique_ptr& other) = delete;
private:
    T* ptr;
};


使用uniquePtr

int main()
{
    unique_ptr<string> p1(new string("hellow"));
    cout << *p1 << endl;
    *p1 = "你好!";
    cout << *p1 << endl;
    return 0;
}


shared_ptr

共享指针将记录有多少个指针共同享有某个对象的所有权。当有更多指针被设置为指向该对象时,引用计数随之增加;当指针和对象分离时,则引用计数也相应减少。当引用计数降低至0时,该对象被删除。


实现SharedPtr

template<class T>
class SharedPtr
{
public:
    SharedPtr(T* ptr = nullptr)
    {
        count = new int(1);
        this->ptr = ptr;
    }
    SharedPtr(SharedPtr& other)
    {
        *other.count += 1;
        count = other.count;
        ptr = other.ptr;
    }
    SharedPtr& operator=(SharedPtr& other)
    {
        *other.count += 1;
        count = other.count;
        ptr = other.ptr;
        return *this;
    }
    ~SharedPtr()
    {
        --(*count);
        cout << *count << "\t" << ptr << endl;
        if (ptr != nullptr && 0 == *count)
        {
            cout << "释放!" << endl;
            delete ptr;
            ptr = nullptr;
            delete count;
            count = nullptr;
        }
    }
    T& operator*() const
    {
        return *ptr;
    }
    T* get() const
    {
        return ptr;
    }
    T* operator->() const
    {
        return ptr;
    }
    int use_count(void) const
    {
        return *count;
    }
private:
    T* ptr;
    int* count;
};


使用shared_ptr

int main()
{
    Shared_ptr<int> p1(new int);
    Shared_ptr<int> p2(p1);
    *p1 = 10;
    cout << *p1 << endl;
    cout << *p2 << endl;
    cout << p1.use_count() << endl;//获取当前引用个数
    return 0;
}

weak_ptr

C++11标准虽然将 weak_ptr 定位为智能指针的一种,但该类型指针通常不单独使用(没有实际用处),只能和shared_ptr类型指针搭配使用。甚至可以将 weak_ptr 类型指针视为 shared_ptr 指针的一种辅助工具,借助weak_ptr类型指针, 我们可以获取 shared_ptr 指针的一些状态信息,比如有多少指向相同的 shared_ptr 指针、shared_ptr 指针指向的堆内存是否已经被释放等等。


使用weak_ptr

class Test
{
public:
    Test()
    {
        cout << "Test()" << endl;
    }
    ~Test()
    {
        cout << "~Test()" << endl;
    }
    void Output() const
    {
        cout << "Output()" << endl;
    }
};
weak_ptr<Test> test()
{
    shared_ptr<Test> p(new Test);
    return p;
}
int main()
{
    weak_ptr<Test> w1 = test();
    shared_ptr<Test> s1 = w1.lock();//检查对象是否被释放
    cout << s1 << endl;
    if (s1)
    {
        cout << "对象依旧存在!" << endl;
    }
    else
    {
        cout << "对象已被释放" << endl;
    }
    return 0;
}

在Effective C++第13条款中也建议:如果想要手工释放资源,容易发送某些错误。使用这类智能指针往往能让程序员更少犯错。

目录
相关文章
|
9月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
301 26
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
10月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
176 1
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
648 68
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
571 0
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
307 3
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
682 0
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
12月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
463 12