calloc与realloc和malloc的区别以及new

 

目录

calloc、realloc 和 malloc

三个函数的区别在于

更详细的示例代码

交叉使用

内存泄漏

悬空指针

内存重叠

new 的语法

使用 new 运算符在堆上创建学生对象的示例

new和malloc都可以用于在堆上分配内存



calloc、realloc 和 malloc

是 C/C++ 中用于动态内存分配的函数。

  1. malloc 函数:malloc 函数用于在堆上分配指定大小的内存块。它接受一个参数,即所需内存块的字节数,返回一个指向分配内存的指针。如果分配失败,则返回 NULL。

  2. calloc 函数:calloc 函数类似于 malloc 函数,但它还会将分配的内存块初始化为零。它接受两个参数,即所需内存块的个数和每个内存块的字节数。它返回一个指向分配内存的指针。如果分配失败,则返回 NULL。

  3. realloc 函数:realloc 函数用于重新分配已分配内存的大小。它接受两个参数,即原始内存块的指针和新的内存块大小。如果内存分配成功,则返回一个指向新分配内存的指针。如果分配失败,则返回 NULL。同时,realloc 函数还会尝试保留原始内存块中的数据,将其复制到新分配的内存块中。

三个函数的区别在于

  • malloc 只负责分配指定字节数的内存块,不对其进行初始化。
  • calloc 在分配内存块的同时,将其初始化为零。
  • realloc 用于重新分配内存块的大小,并尝试保留原始内存块中的数据。

注意:虽然这些函数在 C 中广泛使用,但在 C++ 中,推荐使用 new 和 delete 运算符进行动态内存分配和释放,以便与对象的构造函数和析构函数配合使用,更好地管理内存和资源。

当涉及到内存分配和重新分配时,以下是

更详细的示例代码

#include <iostream>
#include <cstdlib>int main() {// 例子1:使用 malloc 分配内存int* ptr = (int*)malloc(5 * sizeof(int)); // 在堆上分配 5 个 int 类型大小的内存块if (ptr == NULL) {std::cout << "内存分配失败" << std::endl;return 1;}// 例子2:使用 calloc 分配内存并初始化int* ptr2 = (int*)calloc(5, sizeof(int)); // 在堆上分配 5 个 int 类型大小的内存块,并初始化为零if (ptr2 == NULL) {std::cout << "内存分配失败" << std::endl;return 1;}// 例子3:使用 realloc 重新分配内存int* ptr3 = (int*)realloc(ptr, 10 * sizeof(int)); // 将之前分配的内存块大小重新调整为 10 个 int 类型大小if (ptr3 == NULL) {std::cout << "内存分配失败" << std::endl;free(ptr); // 释放之前分配的内存块return 1;}// 释放内存free(ptr);free(ptr2);free(ptr3);return 0;
}

在这个例子中,我们首先使用 malloc 函数分配了一个包含 5 个 int 类型大小的内存块,并将其分配给 ptr 指针。然后,我们使用 calloc 函数分配了另一个包含 5 个 int 类型大小的内存块,并将其初始化为零,并将其分配给 ptr2 指针。最后,我们使用 realloc 函数重新分配了 ptr 指针指向的内存块的大小,将其调整为包含 10 个 int 类型大小,并将其分配给 ptr3 指针。

请注意,在使用这些函数进行内存分配之后,需要使用 free 函数释放已分配的内存,以避免内存泄漏。

交叉使用

malloccallocrealloc 函数时,可能会导致一些问题,如内存泄漏、悬空指针和内存重叠等。下面是这些问题的一些例子:

内存泄漏

如果在分配内存后没有正确释放它,就会导致内存泄漏。这意味着分配的内存将无法再被使用,并且会占用系统资源。例如,在下面的代码中,内存块 ptr 分配后没有被释放:

int* ptr = (int*)malloc(10 * sizeof(int));
// 使用分配的内存块进行操作
// 忘记释放内存块

悬空指针

悬空指针是指指向已释放或未分配内存的指针。当你尝试访问悬空指针时,可能会导致程序崩溃或产生未定义的行为。例如,在下面的代码中,内存块 ptr 被释放后,但指针没有被置为 NULL:

int* ptr = (int*)malloc(10 * sizeof(int));
free(ptr);
// ptr 是一个悬空指针

内存重叠

当你使用 realloc 函数重新分配内存时,如果新的内存块与原始内存块重叠,会导致未定义的行为。这种情况可能会破坏数据或导致程序崩溃。例如,在下面的代码中,当 realloc 函数将原始内存块扩展到更大的大小时,与之前分配的内存块可能发生重叠:

int* ptr = (int*)malloc(5 * sizeof(int));
int* ptr2 = (int*)realloc(ptr, 10 * sizeof(int)); // 内存重叠

为了避免这些问题,建议在使用 malloccallocrealloc 函数时注意以下几点:

  • 确保在分配内存后及时调用 free 函数释放内存。
  • 在释放内存后将指针置为 NULL,以避免悬空指针。
  • 在使用 realloc 函数重新分配内存时,确保新的内存块与原始内存块没有重叠。

此外,C++ 中还提供了更安全和方便的动态内存分配和释放方式,推荐使用 newdelete 运算符进行管理动态内存。

C++ 中,new 运算符用于在动态存储区(堆)中分配指定大小的内存。

new 的语法

pointer = new type;

其中,type 指向要分配内存的类型,可以是基本类型、类类型或数组类型。pointer 是一个指针,它指向新分配的内存块的开头。

例如,下面是一个使用 new 运算符在堆上分配一个整数的示例:

int* ptr = new int;

在上面的示例中,new 运算符返回一个指向分配的内存块的指针,这个内存块的大小与 int 类型相同。我们将这个指针赋值给 ptr 变量,以便稍后可以使用它来访问这个内存块中的数据。

如果要在堆上分配一个数组,可以使用以下语法:

pointer = new type [number_of_elements];

其中,type 是数组元素的类型,number_of_elements 是数组元素的数量。

例如,下面是一个使用 new 运算符在堆上分配一个整数数组的示例:

int* ptr = new int[5];

在上面的示例中,new 运算符返回一个指向分配的内存块的指针,这个内存块的大小等于 5 * sizeof(int) 字节。我们将这个指针赋值给 ptr 变量,以便稍后可以使用它来访问这个内存块中的数据。

除了分配内存之外,new 运算符还可以调用构造函数来初始化新分配的内存。例如,下面是一个使用 new 运算符在堆上分配一个类的示例:

class MyClass {
public:MyClass() {std::cout << "MyClass 构造函数被调用" << std::endl;}
};MyClass* ptr = new MyClass;

在上面的示例中,new 运算符返回一个指向分配的 MyClass 类型内存块的指针,并调用 MyClass 类的默认构造函数进行初始化。我们将这个指针赋值给 ptr 变量,以便稍后可以使用它来访问这个内存块中的数据。

需要注意的是,使用 new 运算符分配的内存必须使用 delete 运算符进行释放。例如,上面的示例应该如下释放内存:

delete ptr;

如果分配的是数组,则应使用以下语法释放内存:

delete[] pointer;

例如,下面是一个使用 delete 运算符释放整数数组的示例:

delete[] ptr;

需要注意的是,如果不释放使用 new 运算符分配的内存,就会导致内存泄漏。在释放内存之后,最好将指针置为 NULL,以避免悬空指针的问题。

当你使用 new 运算符在堆上分配内存时,可以考虑使用类来演示。下面是一个

使用 new 运算符在堆上创建学生对象的示例

#include <iostream>
#include <string>class Student {
public:Student(const std::string& name, int age) : name(name), age(age) {std::cout << "学生对象 " << name << " 被创建" << std::endl;}~Student() {std::cout << "学生对象 " << name << " 被销毁" << std::endl;}void DisplayInfo() {std::cout << "姓名:" << name << std::endl;std::cout << "年龄:" << age << std::endl;}private:std::string name;int age;
};int main() {// 使用 new 运算符在堆上创建一个学生对象Student* ptr = new Student("小明", 18);// 调用对象的成员函数ptr->DisplayInfo();// 释放内存delete ptr;return 0;
}

在上面的示例中,我们定义了一个 Student 类,它有一个带有姓名和年龄参数的构造函数、一个析构函数和一个显示信息的成员函数。在 main 函数中,我们使用 new 运算符在堆上创建一个 Student 对象,并将返回的指针赋给 ptr 变量。然后,我们调用 ptr 指向的对象的成员函数来显示学生的信息。最后,我们使用 delete 运算符释放了分配的内存。

运行上述代码,输出结果如下:

学生对象 小明 被创建
姓名:小明
年龄:18
学生对象 小明 被销毁

这个示例演示了如何使用 newdelete 运算符来动态分配和释放对象的内存。记得在使用完毕后及时释放内存,以避免内存泄漏。

这段代码是一个简单的C++示例,演示了如何使用new运算符在堆上创建对象,并使用delete释放内存。

首先,代码定义了一个名为Student的类,表示学生对象。该类有两个私有成员变量:name(姓名)和age(年龄)。构造函数Student用于初始化对象的姓名和年龄,并在创建对象时打印一条消息。析构函数~Student在对象销毁时被调用,用于释放资源并打印一条销毁消息。成员函数DisplayInfo用于打印学生对象的姓名和年龄。

std::string是C++标准库中的一个类,用于表示字符串。它是C++中处理字符串的一种常用方式。

std::string类提供了各种成员函数和操作符,用于对字符串进行操作,包括字符串的连接、截取、查找、替换等。它还提供了许多便捷的方法,如获取字符串长度、判断字符串是否为空、将字符串转换为整数等。

使用std::string类可以方便地进行字符串的操作和处理,而不需要手动管理内存或处理字符数组。它也提供了一些与C风格字符串的兼容性函数,以便与现有的C代码进行交互。

要在C++程序中使用std::string,需要包含头文件 <string>,并使用 std 命名空间。例如:

#include <string>
#include <iostream>int main() {std::string str = "Hello, World!";std::cout << str << std::endl;return 0;
}

上述代码创建了一个名为 strstd::string 对象,并将其初始化为 "Hello, World!"。然后,通过 std::cout 输出该字符串到控制台。

std::string是C++中用于表示字符串的类,提供了丰富的字符串操作方法和功能。

main函数中,使用new运算符在堆上动态地创建了一个Student对象,并将指向该对象的指针赋值给ptr变量。创建对象时,构造函数会被调用,并打印一条创建消息。

接下来,通过ptr指针调用了对象的成员函数DisplayInfo,以打印学生对象的姓名和年龄。

最后,在程序结束前,使用delete运算符释放了通过new分配的内存。释放内存时,析构函数会被调用,并打印一条销毁消息。

这样,该程序就完成了对堆上动态分配对象的创建、使用和销毁的过程。使用newdelete可以在堆上分配和释放内存,使得对象的生命周期可以由程序控制,更加灵活。

newmalloc都可以用于在堆上分配内存

但它们之间有以下几个区别:

  1. new是C++的关键字,而malloc是C语言的库函数。因此,在C++中使用new更为方便和安全,而在C中则更常用malloc

  2. new会自动计算要分配的内存大小,并返回指向已分配内存的指针。而malloc需要手动计算内存大小,并返回一个void*类型的指针,需要进行类型转换后才能使用。这可能会导致一些潜在的错误。

  3. new会调用类的构造函数来初始化对象,而malloc只是简单地分配一块内存。因此,使用new可以确保对象被正确地初始化,而使用malloc需要手动调用构造函数。

  4. new会抛出异常,而malloc只会返回NULL。在内存分配失败时,new会抛出std::bad_alloc异常,而malloc则会返回NULL指针。因此,在使用new时需要考虑异常处理,而在使用malloc时需要检查返回值是否为NULL

虽然newmalloc都可以用于在堆上分配内存,但由于new更方便、更安全,而且可以自动初始化对象,因此在C++中更常用new。不过,对于一些特殊的内存分配场景,如需要对齐或不需要初始化等情况下,可能需要使用malloc

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/635393.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Mermaid使用教程(绘制各种图)

Mermaid使用教程&#xff08;绘制各种图&#xff09; 文章目录 Mermaid使用教程&#xff08;绘制各种图&#xff09;简介饼状图简单的例子应用案例 序列图简单案例应用案例另一个应用案例 甘特图简单案例应用案例一个更为复杂的应用案例 Git图简单案例 总结 简介 本文将主要介…

Elastic 8.12:AI Assistant for Observability 正式发布,更新至 Apache Lucene 9.9

作者&#xff1a;来自 Elastic Brian Bergholm 今天&#xff0c;我们很高兴地宣布 Elastic 8.12 全面上市。 有哪些新的功能&#xff1f; 8.12 版本的两个最重要的组成部分包括 Elastic AI Assistant for Observability 的 正式发布版 和 Apache Lucene 9.9 的更新&#xff08…

CVE2020-1938漏洞复现

这个漏洞是tomcat的 然后我们先了解漏洞产生的原理 首先我们先来看tmocat纠结是干什么的 tomcat是个中间件 最主要的两个结构、 servlet的定义和部分源码&#xff0c; 漏洞就是从这来的 tomcat处理http请求 源码分析 tomcat 8.5.46 哎 这教学视频讲半天看不懂 不看原…

RAG中的3个高级检索技巧

RAG系统检索的文档可能并不总是与用户的查询保持一致&#xff0c;这是一个常见的现象。当文档可能缺乏查询的完整答案或者包含冗余信息或包含不相关的细节&#xff0c;或者文档的顺序可能与用户的意图不一致时&#xff0c;就会经常出现这种情况。 本文将探讨三种有效的技术来增…

DC-5靶机做题记录

靶机下载地址&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1aZRB-hqvqLlGUmAPFljnIA?pwdelxg 提取码&#xff1a;elxg 参考&#xff1a; DC5官方地址&#xff1a;https://www.vulnhub.com/entry/dc-5,314/DC5靶机地址&#xff1a;https://download.vulnhub.com/d…

2024年17款最好用的免费录制网课软件分享

近年来&#xff0c;在线课程越来越受欢迎&#xff0c;为学习者提供了方便、灵活的教育内容获取方式。随着电子学习和网络培训的兴起&#xff0c;有许多工具可以帮助课程创建者记录和编辑其在线内容。虽然市场上有许多免费和付费工具&#xff0c;但本文将重点介绍 17 个用于录制…

前端学习路线图和一些经验

关于前端目前个人建议的一个路线,也是自己之前前端学习时候的一个大致路线,给想要学习前端的小白一个参考,以前自己刚开始接触前端的时候就是不知道该按照什么路线学习 eg-前端是做什么的&#xff1f; 就是开发网站,移动端&#xff0c;小程序之类的页面 调调接口完成页面的渲…

Ansible详解(架构,模块)及部署示例

Ansible概述 Ansible是一个基于Python开发的配置管理和应用部署工具&#xff0c;也在自动化管理领域大放异彩。它融合了众多老牌运维工具的优点&#xff0c;几乎可以实现Puppet和Saltstack能实现的功能。 Ansible是一款开源的IT自动化工具&#xff0c;它能够自动执行配置管理、…

如何免费从 SD 卡恢复已删除的文件?(照片、视频、MP3)

今天我们将告诉您如何免费从格式化的 SD 卡或闪存卡恢复文件。 特别是如果您是一名摄影师、博主或任何处理内容的人&#xff0c;您的 SD 卡上有一些内容&#xff0c;但您不小心删除或格式化了&#xff0c;现在您要向自己道歉。 无需担心&#xff0c;因为今天我们将告诉您如何…

vue写了debugger谷歌浏览器打开控制台没进断点

vue代码中打了断点&#xff0c;谷歌打开f12进不了断点解决方案如下 1、打开谷歌浏览器控制台&#xff0c;点击设置 2、在 Ignore List 中将“Enable Ignore Listing”勾选去掉&#xff0c;然后就可以正常使用debugger了

为了最大限度利用带宽,传输通道容量如何计算

一、结论&#xff1a; 传送通道容量 带宽 ✖️ 往返时延 二、过程&#xff1a; 1、传播时延和发送时延&#xff08;带宽&#xff09;通常决定一个报文的发送时间。 传输介质确定的时候&#xff0c;传播时延固定&#xff0c;发送时延根据数据报文的大小而不同。 ①传播时延…

GPT应用开发:编写插件获取实时天气信息

欢迎阅读本系列文章&#xff01;我将带你一起探索如何利用OpenAI API开发GPT应用。无论你是编程新手还是资深开发者&#xff0c;都能在这里获得灵感和收获。 本文&#xff0c;我们将继续展示聊天API中插件的使用方法&#xff0c;让你能够轻松驾驭这个强大的工具。 插件运行效…

elasticsearch6.6.0设置访问密码

elasticsearch6.6.0设置访问密码 第一步 x-pack-core-6.6.0.jar第二步 elasticsearch.yml第三步 设置密码 第一步 x-pack-core-6.6.0.jar 首先破解 x-pack-core-6.6.0.jar 破解的方式大家可以参考 https://codeantenna.com/a/YDks83ZHjd 中<5.破解x-pack> 这部分 , 也可…

SQL Server Management Studio创建数据表

文章目录 一、建表注意事项1.1 数据类型1.2 建立数据表的基本SQL语法 二、实例说明2.1 创建数据表2.2 实例2 三、标识列和主键示例&#xff1a; 一、建表注意事项 1.1 数据类型 可以看这个去了解数据类型&#xff1a; 1.2 建立数据表的基本SQL语法 建立数据表的基本 SQL 语…

【正点原子STM32连载】 第四十四章 外部SRAM实验 摘自【正点原子】APM32E103最小系统板使用指南

1&#xff09;实验平台&#xff1a;正点原子APM32E103最小系统板 2&#xff09;平台购买地址&#xff1a;https://detail.tmall.com/item.htm?id609294757420 3&#xff09;全套实验源码手册视频下载地址&#xff1a; http://www.openedv.com/docs/boards/xiaoxitongban 第四…

【备战蓝桥杯】快来学吧~ 图论巩固,Delia的生物考试

蓝桥杯备赛 | 洛谷做题打卡day12 文章目录 蓝桥杯备赛 | 洛谷做题打卡day12最大食物链计数题目背景题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示题解代码总的思路&#xff1a;拓扑排序 我的一些话 最大食物链计数 题目背景 你知道食物链吗&#xff1f;Delia 生…

SSM汽车维修管理系统

工具使用情况&#xff1a; eclipsetomcatmysqljdk 技术架构&#xff1a; 后台&#xff1a;springspring mvcmybatis 前台&#xff1a;easyui 功能介绍&#xff1a; 汽车维修管理、车辆接待、维修项目登记、维修领料、质检完工、消费结算 配件管理、财务管理、基础数据管理…

二进制表示(14)

题目 public class Main {public static String con01(int x,int n) {StringBuffer s new StringBuffer();while(x!0) {s.append(x%n);x/n;}return s.reverse().toString();}public static int con02(int x,int n) {StringBuffer s new StringBuffer();int sum 0;while(x!0…

二、VS2019编译的VTK9.0.0 + Qt 5.14.2 环境测试

1. 使用CMake VS2019 编译vtk 9.0.0 时,需要启用支持Qt开关、如下图 如果不会编译的可以参见我的这篇文章: 一、VTK 9.0.0 编译安装步骤 VS2019 CMake3.26.0-CSDN博客 打开Qt5.14.2 ,创建Qt Widget 项目: 构建设置选择 MSVC2017 64bit pro 项目文件加入两行配置: …

鸿蒙开发系列教程(四)--ArkTS语言:基础知识

1、ArkTS语言介绍 ArkTS是HarmonyOS应用开发语言。它在保持TypeScript&#xff08;简称TS&#xff09;基本语法风格的基础上&#xff0c;对TS的动态类型特性施加更严格的约束&#xff0c;引入静态类型。同时&#xff0c;提供了声明式UI、状态管理等相应的能力&#xff0c;让开…