文章目录
- 1、动态分配内存的应用
- 2、动态分配内存与分配给普通变量的内存有什么不同?
- 3、C++ 中如何分配/释放内存
- 4、new 操作符
- 4.1 使用new的语法
- 4.2 初始化内存
- 4.3 分配内存块
- 4.4 普通数组声明 Vs 使用new
- 4.5 如果运行时没有足够内存可用怎么办?
- 5、delete 操作符
C/ C++中的动态内存分配是指由程序员手动执行内存分配。动态内存分配是在堆上进行,非静态变量和局部变量在栈上分配内存。
1、动态分配内存的应用
- 动态分配内存的一种用途是分配可变大小的内存,这对于编译器分配的内存来说是不可能的,除非是可变长度的数组。
- 最重要的用途是提供给程序员的灵活性。无论何时需要或不再需要,我们都可以自由地分配和释放内存。这种灵活性在很多情况下都很有用。例子有链表、树等。
2、动态分配内存与分配给普通变量的内存有什么不同?
普通变量 int a
,char str[10]
等的内存是自动分配和释放的。对于像“int *p = new int[10]
”这样动态分配的内存,当不再需要内存时,释放内存是程序员的责任。如果程序员不释放内存,就会导致内存泄漏(直到程序终止才释放内存)。
3、C++ 中如何分配/释放内存
C语言使用 malloc()
和 calloc()
函数在运行时动态分配内存,使用 free()
函数释放动态分配的内存。C++ 除了支持这些函数,还提供了 new 和 delete 两个操作符,可以更好、更简单地分配和释放内存。
4、new 操作符
new操作符表示在空闲存储区(Free Store)上分配内存的请求。如果有足够的内存可用,new运算符会初始化内存,并将新分配的已经初始化的内存的地址返回给指针变量。
4.1 使用new的语法
pointer-variable = new data-type;
这里的指针变量是data-type类型的指针。数据类型可以是任何内置数据类型(包括数组),也可以是任何用户定义数据类型(包括结构体和类)。
例子:
// Pointer initialized with NULL
// Then request memory for the variable
int *p = NULL;
p = new int;
OR
// Combine declaration of pointer
// and their assignment
int *p = new int;
4.2 初始化内存
也可以使用new操作符初始化内置数据类型的内存。对于自定义数据类型,需要一个构造函数(输入数据类型)来初始化值。下面是这两种数据类型初始化的例子:
语法:
pointer-variable = new data-type(value);
例子:
int* p = new int(25);
float* q = new float(75.25);
// Custom data type
struct cust
{
int p;
cust(int q) : p(q) {}
cust() = default;
//cust& operator=(const cust& that) = default;
};
int main()
{
// Works fine, doesn’t require constructor
cust* var1 = new cust;
//OR
// Works fine, doesn’t require constructor
var1 = new cust();
// Notice error if you comment this line
cust* var = new cust(25);
return 0;
}
4.3 分配内存块
还可以使用new运算符来分配一个类型为data type 的内存块(一个数组)
语法:
pointer-variable = new data-type[size];
其中 size
(变量) 指定了数组中元素的个数。
例子:
int *p = new int[10]
动态地为连续10个 int 类型的整数分配内存,返回一个指向数组的第一个元素的指针,该元素被赋值为 p
(指针)。p[0]
表示第1个元素,p[1]
表示第2个元素,以此类推。
4.4 普通数组声明 Vs 使用new
普通数组声明和使用 new 分配内存块是有区别的。最重要的区别是,普通数组是由编译器释放内存的(如果数组是局部的,那么函数返回或完成时释放)。但是,动态分配内存的数组只有当程序员释放或程序终止时才会释放,否则就一直占用该内存。
4.5 如果运行时没有足够内存可用怎么办?
如果堆中没有足够内存可供分配,new 操作就会通过抛出一个 类型为std::bad_alloc
异常来表示失败,除非 new 操作符使用了 “nothrow”,这种情况下它会返回一个 NULL 指针。因此,在使用 new 创建的指针前最好先进行检查。
int *p = new(nothrow) int;
if (!p)
{
cout << "Memory allocation failed\n";
}
5、delete 操作符
因为释放动态分配的内存是程序员的责任,所以C++语言为程序员提供了 delete 操作符。
语法:
// Release memory pointed by pointer-variable
delete pointer-variable;
此处的指针变量指向的是由 new 创建的数据对象。
例子:
delete p;
delete q;
要释放指针变量指向的动态分配内存的数组,可以使用以下形式的delete:
// Release block of memory
// pointed by pointer-variable
delete[] pointer-variable;
Example:
// It will free the entire array
// pointed by p.
delete[] p;
完整例子源码:
// C++ program to illustrate dynamic allocation
// and deallocation of memory using new and delete
#include <iostream>
using namespace std;
int main ()
{
// Pointer initialization to null
int* p = NULL;
// Request memory for the variable
// using new operator
p = new(nothrow) int;
if (!p)
cout << "allocation of memory failed\n";
else
{
// Store value at allocated address
*p = 29;
cout << "Value of p: " << *p << endl;
}
// Request block of memory
// using new operator
float *r = new float(75.25);
cout << "Value of r: " << *r << endl;
// Request block of memory of size n
int n = 5;
int *q = new(nothrow) int[n];
if (!q)
cout << "allocation of memory failed\n";
else
{
for (int i = 0; i < n; i++)
q[i] = i+1;
cout << "Value store in block of memory: ";
for (int i = 0; i < n; i++)
cout << q[i] << " ";
}
// freed the allocated memory
delete p;
delete r;
// freed the block of allocated memory
delete[] q;
return 0;
}
输出:
Value of p: 29
Value of r: 75.25
Value store in block of memory: 1 2 3 4 5
时间复杂度: O ( n ) O(n) O(n), n n n给定的内存大小。
注意:
- delete 可以释放空指针
int *ptr = NULL;
//Work fine!
delete ptr;
- 同一个指针不能使用两次delete,这是未定义行为
#include<iostream>
using namespace std;
int main()
{
int *ptr = new int;
delete ptr;
//为同一个指针调用两次delete是未定义的行为,任何情况都可能发生,程序可能会crash也可能没有任何问题
delete ptr;
return 0;
}
翻译自文章 new and delete Operators in C++ For Dynamic Memory