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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

链表的相交

链表的相交 力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台备战技术面试&#xff1f;力扣提供海量技术面试资源&#xff0c;帮助你高效提升编程技能&#xff0c;轻松拿下世界 IT 名企 Dream Offer。https://leetcode.cn/problems/intersection-of-tw…

【NVIDIA】Jetson Orin Nano系列:安装 Qt6、firefox、jtop、flameshot

1、使用命令安装 sudo apt install qtcreator sudo apt install qt6-* sudo apt install libqt6* sudo apt install qml-qt6 sudo apt install qmlscene-qt6 sudo apt install assistant-qt6 sudo apt install designer-qt62、启动 qtcreator 3、常用工具安装 sudo apt in…

ros2学习笔记-CLI工具,记录命令对应操作。

目录 环境变量turtlesim和rqt以初始状态打开rqt node启动节点查看节点列表查看节点更多信息命令行参数 --ros-args topic话题列表话题类型话题列表&#xff0c;附加话题类型根据类型查找话题名查看话题发布的数据查看话题的详细信息查看类型的详细信息给话题发布消息&#xff0…

线程状态转换

&#x1f4dd;个人主页&#xff1a;五敷有你 &#x1f525;系列专栏&#xff1a;并发编程⛺️稳中求进&#xff0c;晒太阳 程状态转换 假设有线程Thread t 情况1 new-->RUNNABLE 当调用t.start()方法时&#xff0c;由new ->RUNNABLE 情况2 RUNNABLE WAITING t…

无/自监督去噪(1)——一个变迁:N2N→N2V→HQ-SSL

目录 1. 前沿2. N2N3. N2V——盲点网络&#xff08;BSNs&#xff0c;Blind Spot Networks&#xff09;开创者3.1. N2V实际是如何训练的&#xff1f; 4. HQ-SSL——认为N2V效率不够高4.1. HQ-SSL的理论架构4.1.1. 对卷积的改进4.1.2. 对下采样的改进4.1.3. 比N2V好在哪&#xff…

c++基础2

一、c的引用 引用和指针的的区别&#xff1f; 引用是一种更安全的指针&#xff1a; 1. 引用必须初始化&#xff0c;指针可以不用初始化 int a 10; int *p; // 指针可能是野指针 int &b a;//引用赋值"&#xff0c;通常指的是直接修改引用所引用的对象的值&#xff0…

Three.JS教程1 环境搭建、场景与相机

Three.JS教程1 环境搭建、场景与相机 一、Three.JS简介二、环境搭建1. 开发准备2. 安装 three.js3. 新建文件index.htmlmain.js 4. 关于附加组件5. 启动 三、创建场景1. 场景的概念2. 相机的概念3. 相机的几个相关概念&#xff08;1&#xff09;视点&#xff08;Position&#…

信息登记小程序怎么做_重塑用户互动,开启全新营销篇章

信息登记小程序&#xff1a;重塑用户互动&#xff0c;开启全新营销篇章 在数字化浪潮中&#xff0c;小程序以其便捷、高效的特点&#xff0c;逐渐成为企业与用户之间沟通的桥梁。其中&#xff0c;信息登记小程序更是凭借其独特的定位&#xff0c;在众多小程序中脱颖而出。本文…

荣誉艾尔迪亚人的题解

目录 原题描述&#xff1a; 题目背景 题目描述 输入格式 输出格式 样例 Input 1 Output 1 Input 2 Output 2 数据范围&#xff1a; 样例解释 主要思路&#xff1a; 代码code&#xff1a; 原题描述&#xff1a; 时间限制: 1000ms 空间限制: 65536kb 题目背景 ​…

ros2 基础教程-使用ROS 2进行相机标定

ROS 2进行相机标定&#xff08;Camera Calibration&#xff09; 相机&#xff08;摄像头&#xff09;是一种非常精密的光学仪器&#xff0c;对外界环境的感知非常敏感。由于摄像头内部和外部的一些原因&#xff0c;摄像头采集的图像常常会发生一定的畸变。如果直接将采集到的图…

【分布式技术】ELK大型日志收集分析系统

目录 步骤一&#xff1a;完成JAVA环境部署 步骤二&#xff1a;部署ES节点&#xff08;三台主机&#xff09; 步骤三&#xff1a;内核参数修改 步骤四&#xff1a;web端查看验证 步骤五&#xff1a;yum安装nginx 步骤六&#xff1a;完成logstash部署 步骤七&#xff1a;部…

matlab抽取与插值

什么是抽取&#xff1f; 我们假设一个数字信号 x ( n ) , n 1 , 2 , . . . , N x(n),n1,2,...,N x(n),n1,2,...,N共有 N N N个点&#xff0c;抽取就是每个几个点抽1个点&#xff0c;比如2倍抽取&#xff0c;那么抽取后的信号为 y ( n ) , y ( 1 ) x ( 1 ) , y ( 2 ) x ( 3 …

stm32 FOC 电机介绍

今年开始学习foc控制无刷电机&#xff0c;这几天把所学整理一下&#xff0c;记录一下知识内容。 前言: 为什么要学习FOC? 1.电机控制是自动化控制领域重要一环。 2.目前直流无刷电机应用越来越广泛&#xff0c;如无人机、机械臂、云台、仿生机器人等等。 需要什么基础&…

项目管理十大知识领域之风险管理

1. 项目风险管理的定义与概述 项目风险管理是指为了实现项目目标&#xff0c;有计划地识别、评估和应对项目中的各种风险的过程。项目风险管理的核心在于提前辨识到可能对项目目标产生不利影响的不确定因素&#xff0c;并采取适当的措施降低或消除这些风险&#xff0c;以保障项…

three.js从入门到精通系列教程005 - three.js使用鼠标拖拽缩放浏览全景图

<!DOCTYPE html> <html><head><meta charset"UTF-8"><title>three.js从入门到精通系列教程005 - three.js使用鼠标拖拽缩放浏览全景图</title><script src"ThreeJS/three.js"></script><script src&qu…

基于SpringBoot Vue博物馆管理系统

大家好✌&#xff01;我是Dwzun。很高兴你能来阅读我&#xff0c;我会陆续更新Java后端、前端、数据库、项目案例等相关知识点总结&#xff0c;还为大家分享优质的实战项目&#xff0c;本人在Java项目开发领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#x…

深入数仓离线数据同步:问题分析与优化措施

一、前言 在数据仓库领域&#xff0c;离线数仓和实时数仓是常见的两种架构类型。离线数仓一般通过定时任务在特定时间点&#xff08;通常是凌晨&#xff09;将业务数据同步到数据仓库中。这种方式适用于对数据实时性要求不高&#xff0c;更侧重于历史数据分析和报告生成的场景…

Spring第六天(注解开发第三方Bean)

注解开发管理第三方Bean 显然&#xff0c;我们无法在第三方Bean中写入诸如service这样的注解&#xff0c;所以&#xff0c;Spring为我们提供了Bean这一注解来让我们通过注解管理第三方Bean 第二种导入方式由于可读性太低&#xff0c;故只介绍第一种导入方式&#xff0c;这里我…

【JavaEE】线程安全的集合类

作者主页&#xff1a;paper jie_博客 本文作者&#xff1a;大家好&#xff0c;我是paper jie&#xff0c;感谢你阅读本文&#xff0c;欢迎一建三连哦。 本文于《JavaEE》专栏&#xff0c;本专栏是针对于大学生&#xff0c;编程小白精心打造的。笔者用重金(时间和精力)打造&…

【ARM Cortex-M 系列 1.1 -- Cortex-M33 与 M4 差异 详细介绍】

请阅读【嵌入式开发学习必备专栏 之 Cortex-Mx 专栏】 文章目录 背景Cortex-M33 与 M4 差异Cortex-M33Cortex-M4关系和差异举例说明 背景 在移植 RT-Thread 到 瑞萨RA4M2&#xff08;Cortex-M33&#xff09;上时&#xff0c;遇到了hardfault 问题&#xff0c;最后使用了Cortex…