IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 动态内存分配:malloc()/free()、new/delete详解 -> 正文阅读

[C++知识库]动态内存分配:malloc()/free()、new/delete详解

1.内存布局

在 C++ 中,我们把内存详细分为 5 个区域:

  • 栈:一般函数内的局部变量都会放在这里,由编译器自动分配和释放。
  • 堆:程序员通过 malloc()/free() 或 new/delete 来分配和释放。
  • 全局区/静态存储区:存放全局变量和静态变量,程序结束时由系统释放。
  • 常量存储区
  • 程序代码区

堆和栈的区别:

  • 栈:空间是有限的,分配速度快,程序员控制不了。
  • 堆:只要不超出实际拥有的物理内存,同时也在操作系统允许能够分配的最大内存大小之内,都可以分配成功。分配速度比栈慢,程序员可以随时用 malloc()/free() 或 new/delete 来分配和释放,非常灵活。

2.malloc()/free() 的用法

在程序运行过程,为了能够自由控制内存的生命周期、大小,会经常使用堆空间的内存。

在 C 语言中,用 malloc() 和 free() 从堆中分配和释放内存,malloc() 和 free() 是函数。

一般使用格式:

void *malloc(int NumBytes),其中 NumBytes 表示要分配的字节数。分配成功则返回指向被分配内存的指针,分配失败则返回 NULL。当分配的这段内存不再使用的时候,应该用 free() 函数将内存释放掉,供其他地方使用。

void free(void *FirstByte),将之前用 malloc() 分配的内存空间还给程序(操作系统),也就是说释放了这块内存,这样这块内存就被系统回收并在需要的时候由系统分配出去再给其他地方使用。

举例1:

#include<iostream>
using namespace std;

int main()
{
	int* p = NULL;
	p = (int*)malloc(sizeof(int)); // 在堆中分配4个字节

	if (p != NULL)
	{
		*p = 10;
		cout << *p << endl; // 10
		free(p);
	}

	return 0;
}

在这里插入图片描述

举例2:

#include<iostream>
using namespace std;

int main()
{
	int* p = (int*)malloc(100 * sizeof(int)); // 分配能放得下100个整数的内存空间
	
	if (p != NULL)
	{
		int* q = p;
		*q++ = 1;
		*q++ = 5;
		cout << *p << endl; // 1
		cout << *(p + 1) << endl; // 5
		free(p);
	}

	return 0;
}

初始化:

memset() 函数是将较大的数据结构(比如对象、数组等)内存清零的比较快的方法。

#include <iostream>
using namespace std;

int main()
{
	int size = sizeof(int) * 10;

	int* p = (int*)malloc(size);

	// 从p地址开始的连续4个字节中的每一个字节都设置为1
	// 00000001 00000001 ... 00000001 00000001
	memset(p, 1, size);

	cout << *p << endl; // 16843009

	return 0;
}

注意:通过 malloc()/free() 分配和释放的对象不会调用其构造函数和析构函数。

#include <iostream>
using namespace std;

struct Person
{
	int m_age;

	Person()
	{
		cout << "Person::Person()" << endl;
	}

	~Person()
	{
		cout << "Person::~Person()" << endl;
	}

	void run()
	{
		cout << "Person::run() - " << m_age << endl;
	}
};

int main()
{
	Person* p = (Person*)malloc(sizeof(Person));
	p->m_age = 10;
	p->run();
	free(p);

	return 0;
}

在这里插入图片描述

3.new/delete 的用法

new/delete 是关键字或运算符,不是函数。在 C++ 中使用 new 和 delete 在堆中分配和释放内存,不再使用 malloc() 和 free() 来分配和释放内存。

malloc()/free() 和 new/delete 这两对都用于动态地在堆中分配和释放内存,但是 new/delete 比 malloc()/free() 干了更多的事情。new/delete 具备对堆上所分配内存进行初始化和释放的能力,而这些能力是 malloc()/free() 所不具备的。

一般使用格式:

(1) 指针变量名 = new 类型标识符;

(2) 指针变量名 = new 类型标识符(初始值);

(3) 指针变量名 = new 类型标识符[内存单元个数];

初始化:

在这里插入图片描述

举例1:

#include<iostream>
using namespace std;

int main()
{
	int* p = new int(18);

	if (p != NULL)
	{
		cout << *p << endl; // 18
		delete p; // 释放单个int的空间
	}

	return 0;
}

举例2:

#include<iostream>
using namespace std;

int main()
{
	int* pa = new int[100]; // 开辟一个大小为100的整形数组空间

	if (pa != NULL)
	{
		int* q = pa;
		*q++ = 12;
		*q++ = 18;
		cout << *pa << endl; // 12
		cout << *(pa + 1) << endl; // 18

		// 释放pa数组空间。new的时候用了[],那么delete就必须用[],delete[]不用写数组大小
		delete[] pa;
	}

	return 0;
}

注意:delete 回收堆空间内存,表示这块堆空间内存可以重新被别人使用,但这块堆空间并不会被清零,指针也不会被置成 NULL。

#include <iostream>
using namespace std;

int main()
{
	int* p = new int;
	*p = 10;
	delete p; // 堆空间的这4个字节不会被清零,p也不会被置成NULL
	p = nullptr; // 这是个好习惯,表明该指针不指向任何对象了

	return 0;
}

注意:通过 new/delete 分配和释放的对象会调用其构造函数和析构函数。

#include <iostream>
using namespace std;

struct Person
{
	int m_age;

	Person()
	{
		cout << "Person::Person()" << endl;
	}

	~Person()
	{
		cout << "Person::~Person()" << endl;
	}

	void run()
	{
		cout << "Person::run() - " << m_age << endl;
	}
};

int main()
{
	Person* p = new Person; // 构造函数被调用
	p->m_age = 10;
	p->run();
	delete p; // 析构函数被调用

	return 0;
}

在这里插入图片描述

4.new/delete 探秘

在这里插入图片描述

4.1 operator new() 和 operator delete()

operator new() 和 operator delete() 是函数。

new 干了两个事情:

  1. 在堆中分配内存:通过 operator new() 来分配内存
  2. 调用构造函数来初始化内存

delete 也干了两个事:

  1. 调用析构函数
  2. 释放内存:调用 operator delete() 来释放内存

4.2 new 记录分配的内存大小供 delete 使用

new 如何记录分配的内存大小供 delete 使用?

答:不同的编译器,new 内部有不同的实现方式。

int *p = new int; // 4字节
delete p;

删除的时候,编译器怎么知道要回收的是 4 字节?

答:new 内部有记录机制,记录了分配出去多少内存。

4.3 new[]/delete[] 申请和释放一个数组

基本数据类型(内置类型)

举例1:

#include <iostream>
using namespace std;

int main()
{
	int* p = new int(100); // 如果不释放,会泄漏4字节
	delete p;

	return 0;
}

举例2:

#include <iostream>
using namespace std;

int main()
{
	int* p = new int[2]; // 如果不释放,会泄漏8字节
	delete[] p;

	return 0;
}

自定义类型(类类型)

举例1:

#include <iostream>
using namespace std;

class A
{
public:
	A()
	{
		cout << "A::A()" << endl;
	}
	~A()
	{
		cout << "A::~A()" << endl;
	}
};

int main()
{
	A* p = new A(); // 如果不释放,会泄漏1字节
	delete p;

	return 0;
}

在这里插入图片描述

#include <iostream>
using namespace std;

class A
{
public:
	A()
	{
		cout << "A::A()" << endl;
	}
	~A()
	{
		cout << "A::~A()" << endl;
	}
};

int main()
{
	A* p = new A[2](); // 如果不释放,会泄漏6字节
	delete[] p;

	return 0;
}

在这里插入图片描述

疑问:为什么给类型 A 对象数组动态分配内存时多出来 4 个字节,而给内置类型 int 数组动态分配内存时并没有多出来 4 字节?

在上面的程序中,对于类类型 A,调用了两次构造函数、两次析构函数,delete一个数组时,要为每一个数组元素调用析构函数。

但是,对于 delete 表达式,它并不知道数组的元素个数,只有 operator new() 函数和 operator delete() 函数知道。因此,必须有一种手段来告诉 delete 表达式的数组大小是多少。

那么,一种可行的方式就是,多分配一个大小为 4 字节的空间来记录数组大小,同时可以约定前 4 字节来记录大小。那么,由 operator new() 函数分配的地址与 new 表达式返回的地址应该相差 4 个字节。

当然,对于非类类型数组和不需要调用析构函数的类类型数组,这多余的 4 字节就不需要了。

4.4 new/delete、new[]/delete[] 要配对使用

举例1:

#include <iostream>
using namespace std;

int main()
{
	int* pi = new int[3]; // 如果不释放,会泄漏12字节
	//delete pi; // 即使不用delete[]释放,也不会发生内存泄漏	
	delete[] pi; // 这种释放方法才是最规范的

	return 0;
}

举例2:

#include<iostream>
using namespace std;

class A
{
public:
	A()
	{
		cout << "A::A()" << endl;
	}

	// 没有自定义的析构函数
};

int main()
{
	A* pa = new A[2](); // 如果不释放,会泄漏2字节
	//delete pa; // 即使不用delete[]释放,也不会发生内存泄漏
	delete[] pa; // 这种释放方法才是最规范的

	return 0;
}

举例3:

#include <iostream>
using namespace std;

class A
{
public:
	A()
	{
		cout << "A::A()" << endl;
	}
	~A()
	{
		cout << "A::~A()" << endl;
	}
};

int main()
{
	A* pa = new A[2](); // 如果不释放,会泄漏6字节
	//delete pa; // 这里如果不用delete[]释放,系统就会报告异常
	delete[] pa; // 这种释放方法才是最规范的

	return 0;
}

在这里插入图片描述

为什么自己一提供析构函数,不用 delete[] 来释放 new[] 出来的内存就报异常呢?

首先只调用了 1 次 A 的析构函数而不是 2 次,表示肯定有内存泄漏。真正释放内存的是 operator delete() 函数,而多出来的 4 个字节导致释放内存空间错乱。

结论:如果一个对象(内置对象、类对象),使用 new[] 来分配内存,却用单独的 delete(而不是 delete[])来释放内存,那么这个对象需要满足的条件是:对象的类型要么是内置类型或者无自定义的析构函数的类类型。

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-04-09 18:05:39  更:2022-04-09 18:07:38 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 0:17:16-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码