C++中用于动态内存的new和delete操作符

news/2025/2/14 3:03:02/

文章目录

  • 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 achar 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 constructorcust* var1 = new cust;//OR// Works fine, doesn’t require constructorvar1 = new cust();// Notice error if you comment this linecust* 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 nullint* p = NULL;// Request memory for the variable// using new operatorp = 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 operatorfloat *r = new float(75.25);cout << "Value of r: " << *r << endl;// Request block of memory of size nint 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 memorydelete p;delete r;// freed the block of allocated memorydelete[] 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


http://www.ppmy.cn/news/1255561.html

相关文章

【hacker送书活动第7期】Python网络爬虫入门到实战

第7期图书推荐 内容简介作者简介大咖推荐图书目录概述参与方式 内容简介 本书介绍了Python3网络爬虫的常见技术。首先介绍了网页的基础知识&#xff0c;然后介绍了urllib、Requests请求库以及XPath、Beautiful Soup等解析库&#xff0c;接着介绍了selenium对动态网站的爬取和S…

【CTA认证】Android8实现android6以下的应用运行时也要申请权限

需求 CTA入网认证&#xff0c;要求低版本比如Android6以下的应用&#xff0c;运行时&#xff0c;也需要有运行时权限(Runtime Permission)功能&#xff0c;不能默认就取到权限&#xff0c;必须人工在设置中打开才可。 环境 Android 8 实现 frameworks 修改思路是所有APP都…

拼多多财报解读:连接高质量供给与全球消费者,多多跨境动力澎湃

全面拥抱“高质量发展”新阶段后&#xff0c;拼多多再次交出一份惊人的成绩单。 11月28日&#xff0c;拼多多集团发布截至9月30日的2023年第三季度业绩报告。财报显示&#xff0c;拼多多集团今年第三季度收入为688.4亿元&#xff0c;同比增长93.9%&#xff1b;实现美国通用会计…

【Unity动画】为一个动画片段添加事件Events

动画不管播放到那一帧&#xff0c;我们都可以在这里“埋伏”一个事件&#xff08;调用一个函数并且给函数传递一个参数&#xff0c;参数在外部设置&#xff0c;甚至传递一个物体&#xff09;&#xff01; 嗨&#xff0c;亲爱的Unity小伙伴们&#xff01;你是否曾想过为你的动画…

工作几年了,你真的懂 Redis 嘛?

大家好&#xff0c;我是伍六七。一个专注于输出 AI 编程内容的在职大厂资深程序员&#xff0c;全国最大 AI 付费社群破局初创合伙人&#xff0c;关注我一起破除 35 诅咒。 Redis 基本上是大部分技术公司都会使用的缓存框架&#xff0c;但是我发现很多程序员其实并不懂 Redis。 …

Mac下更新python

1. 查看python版本 python3 -V2. 更新python版本 brew upgrade python3. 查看python3的路径&#xff1a; which python34. 修改.bash_profile文件 vim ~/.bash_profile插入python默认执行文件 alias python"/usr/local/bin/python3"5. 应用环境变量 source ~/.…

在线 SQL 模拟器SQL Fiddle使用简介

在线 SQL 模拟器SQL Fiddle使用简介 本文可作为“SQL语言与SQL在线实验工具的使用” https://blog.csdn.net/cnds123/article/details/115038700 一文的补充。 有时候&#xff0c;我们想去验证 SQL语句&#xff0c;却缺少数据库环境&#xff0c;那该怎么办呢&#xff1f; 这…

React使用TailwindCSS

React中使用TailwindCSS TailwindCSS是 下载及初始化 可以查看官网对照自己使用的框架进行配置 npm install -D tailwindcss postcss autoprefixer下载完毕后执行如下命令 npx tailwindcss init -p可以发现项目中多了两个文件 其中默认已经进行了配置&#xff0c;我们需要将…