从C语言到C++_36(智能指针RAII)auto_ptr+unique_ptr+shared_ptr+weak_ptr(上)

简介: 从C语言到C++_36(智能指针RAII)auto_ptr+unique_ptr+shared_ptr+weak_ptr

55a8395061124fd49ce1b7a678e4ff39.png 610a98569f5e465b90378c5e48745524.png 610a98569f5e465b90378c5e48745524.png 1. 智能指针的引入_内存泄漏

为什么需要智能指针?上一篇:

1.1 内存泄漏

上面是异常安全导致的内存泄漏问题,开空间没有释放也可能导致内存泄漏。

什么是内存泄漏?:


       内存泄漏指因为疏忽或错误(逻辑错误)造成程序未能释放已经不再使用的内存的情况。内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存的控制(指针丢了),因而造成了内存的浪费。


       内存泄漏的危害:长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死。

void MemoryLeaks()
{
  int* p1 = (int*)malloc(sizeof(int)); // 1.内存申请了忘记释放
  int* p2 = new int;
 
  int* p3 = new int[10]; // 2.异常安全问题
 
  Func(); // 这里如果Func函数抛异常n,会导致下一行 delete[] p3未执行,p3没被释放.
  delete[] p3;
}

内存泄漏分类(了解):

C/C++程序中一般我们关心两种方面的内存泄漏:

堆内存泄漏(Heap leak):

       堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉。假设程序的设计错误导致这部分内存没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap Leak。

系统资源泄漏:

       指程序使用系统分配的资源,比方套接字、文件描述符、管道等没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行不稳定。

1.2 如何避免内存泄漏

1. 工程前期良好的设计规范,养成良好的编码规范,申请的内存空间记着匹配的去释放。ps:这个理想状态。但是如果碰上异常时,就算注意释放了,还是可能会出问题。需要下一条智能指针来管理才有保证。

2. 采用RAII思想或者智能指针来管理资源。

3. 有些公司内部规范使用内部实现的私有内存管理库。这套库自带内存泄漏检测的功能选项。

4. 出问题了使用内存泄漏工具检测,如Valgrind和Sanitizer。ps:不过很多工具都不够靠谱,或者收费昂贵。

总结 :内存泄漏非常常见,解决方案分为两种:


1. 事前预防型。如智能指针等。


2. 事后查错型。如泄漏检测工具。

2. RAII思想

RAII:是英文Resource Acquisition Is Initialization(资源获取即初始化)的首字母,是一种利用对象生命周期来控制程序资源的简单技术。

这些资源可以是内存,文件句柄,网络连接,互斥量等等。

       RAII:在对象构造时获取资源,接着控制对资源的访问使之在对象的生命周期内始终保持有效,最后在对象析构的时候释放资源。借此,我们实际上把管理一份资源的责任托管给了一个对象。


这种做法有两大好处:

① 不需要显式地释放资源。

② 采用这种方式,对象所需的资源在其生命期内始终保持有效。

2.1 RAII解决异常安全问题

利用RAII思想设计delete资源的类:

#include <iostream>
using namespace std;
double Division(int a, int b)
{
  if (b == 0)
  {
    throw "Divide by Zero Error";
  }
  else
  {
    return ((double)a / (double)b);
  }
}
// 利用RAII思想设计delete资源的类
template<class T>
class SmartPtr
{
public:
  SmartPtr(T* ptr)
    :_ptr(ptr)
  {}
  ~SmartPtr()
  {
    cout << "delete: " << _ptr << endl;
    delete _ptr;
  }
protected:
  T* _ptr;
};
 
void Func()
{
  //1、如果p1这里new 抛异常会如何?
  //2、如果p2这里new 抛异常会如何?
  //3、如果div调用这里又会抛异常会如何?
  //int* p1 = new int;
  //int* p2 = new int;
  //cout << Division() << endl;
  //delete p1;
  //delete p2;
  //cout << "释放资源" << endl; 
 
  SmartPtr<int> sp1(new int);
  SmartPtr<int> sp2(new int);
 
  cout << Division(3, 0) << endl;
}
 
int main()
{
  try
  {
    Func();
  }
  catch (const char* errmsg)
  {
    cout << errmsg << endl;
  }
  catch (...)
  {
    cout << "unknown exception" << endl;
  }
  cout << "return 0;" << endl;
  return 0;
}

运行:

把 Division(3, 0) 改为 Division(3, 1):


2.2 智能指针原理

       上述的SmartPtr还不能将其称为智能指针,因为它还不具有指针的行为。指针可以解引用,也可以通过->去访问所指空间中的内容,因此:AutoPtr模板类中还得需要将* 、->重载下,才可让其像指针一样去使用。

// 1、利用RAII思想设计delete资源的类
// 2、重载operator*和opertaor->,具有像指针一样的行为。
// 3、浅拷贝问题(析构两次,下面讲
template<class T>
class SmartPtr
{
public:
  SmartPtr(T* ptr)
    :_ptr(ptr)
  {}
  ~SmartPtr()
  {
    cout << "delete: " << _ptr << endl;
    delete _ptr;
  }
 
  T& operator*()
  {
    return *_ptr;
  }
 
  T* operator->()
  {
    return _ptr;
  }
protected:
  T* _ptr;
};

       所谓RAII,就是将资源的生命周期和对象的生命周期绑定。从构造函数开始,到析构函数结束。智能指针就是使用了RAII技术,并且利用对象生命周期结束时,编译器会自动调用对象的析构函数来释放资源。


       智能指针的智能就在于资源会被自动释放,不需要显式地释放资源。采用智能指针,对象所需的资源在其生命周期内始终保持有效。


总结智能指针的原理:


1、利用RAII思想设计delete资源的类


2、重载operator*和opertaor->,具有像指针一样的行为。


3、拷贝问题(不同的智能指针的解决方式不一样)

3. auto_ptr

C++98就已经提供了这样的一个智能指针:(注意到上面写着deprecated不推荐使用了)


让上面写的SmartPtr使用编译器自动生成的拷贝构造函数:

#include <iostream>
using namespace std;
// 1、利用RAII思想设计delete资源的类
// 2、重载operator*和opertaor->,具有像指针一样的行为。
// 3、浅拷贝问题
template<class T>
class SmartPtr
{
public:
  SmartPtr(T* ptr)
    :_ptr(ptr)
  {}
  ~SmartPtr()
  {
    cout << "delete: " << _ptr << endl;
    delete _ptr;
  }
 
  T& operator*()
  {
    return *_ptr;
  }
 
  T* operator->()
  {
    return _ptr;
  }
protected:
  T* _ptr;
};
 
int main()
{
  SmartPtr<int> sp1(new int);
  SmartPtr<int> sp2(sp1);
 
  return 0;
}



image.png

上面代码在运行时报错。


       智能指针ap2拷贝复制了ap1,此时ap1和ap2都指向同一块动态内存空间。当程序执行结束以后,ap1对象和ap2对象都会销毁,并且会执行各自的析构函数,所以那份动态空间就会被释放两次,所以报错了。怎么解决?:

       显式定义一个拷贝构造函数,不能让两个智能指针指向同一份动态内存空间。(但是这样没有很好的解决问题,auto_ptr就是这样设计的)

//auto_ptr
#include <iostream>
using namespace std;
// 1、利用RAII思想设计delete资源的类
// 2、重载operator*和opertaor->,具有像指针一样的行为。
// 3、浅拷贝问题
template<class T>
class SmartPtr
{
public:
  SmartPtr(T* ptr)
    :_ptr(ptr)
  {}
  ~SmartPtr()
  {
    cout << "delete: " << _ptr << endl;
    delete _ptr;
  }
  SmartPtr(SmartPtr<T>& ptr)
    :_ptr(ptr._ptr)
  {
    ptr._ptr = nullptr;
  }
 
  T& operator*()
  {
    return *_ptr;
  }
 
  T* operator->()
  {
    return _ptr;
  }
protected:
  T* _ptr;
};
 
int main()
{
  SmartPtr<int> sp1(new int);
  SmartPtr<int> sp2(sp1);
 
  return 0;
}

921fcb1b710c4b1685c01a2f784ab7e3.png

增加一个名字叫A的类重复上面操作:

class A
{
public:
  ~A()
  {
    cout << "~A()" << endl;
  }
protected:
  int _a1 = 0;
  int _a2 = 0;
};
 
int main()
{
  SmartPtr<A> sp1(new A);
  SmartPtr<A> sp2(sp1);
 
  return 0;
}

使用一下库里的auto_ptr试一下:

class A
{
public:
  ~A()
  {
    cout << "~A()" << endl;
  }
protected:
  int _a1 = 0;
  int _a2 = 0;
};
 
int main()
{
  //SmartPtr<A> sp1(new A);
  //SmartPtr<A> sp2(sp1);
  auto_ptr<A> sp1(new A);
  auto_ptr<A> sp2(sp1);
 
  return 0;
}

c71e8d8372574b3c8d40141260186a9c.png

和显式定义一个拷贝构造函数的效果一样。auto_ptr到这种情况就崩了:

class A
{
public:
  ~A()
  {
    cout << "~A()" << endl;
  }
//protected:
  int _a1 = 0;
  int _a2 = 0;
};
 
int main()
{
  //SmartPtr<A> sp1(new A);
  //SmartPtr<A> sp2(sp1);
  auto_ptr<A> sp1(new A);
  auto_ptr<A> sp2(sp1);
  sp1->_a1++;
  sp1->_a2++;
 
  return 0;
}


image.png

从C语言到C++_36(智能指针RAII)auto_ptr+unique_ptr+shared_ptr+weak_ptr(中):https://developer.aliyun.com/article/1522496

目录
相关文章
|
2月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
56 0
|
20天前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
73 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
20天前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
44 9
|
20天前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
40 7
|
1月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
103 13
|
23天前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
23天前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
74 3
|
24天前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
1月前
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
52 11
|
23天前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
35 1