C/C++内存管理(含C++中new和delete的使用)

文章目录

  • C/C++内存管理(含C++中new和delete的使用)
    • 1、C/C++内存分布
    • 2、C语言中动态内存管理方式:malloc/calloc/realloc/free
    • 3、C++动态内存管理
      • 3.1、new/delete操作内置类型
      • 3.2、new/delete操作自定义类型
    • 4、operator new与operator delete函数
    • 5、new和delete的实现原理
      • 5.1、内置类型
      • 5.2、自定义类型
    • 6、定位new表达式(placement-new)

img

C/C++内存管理(含C++中new和delete的使用)

1、C/C++内存分布

我们先来看下面的一段代码和相关问题。

int globalVar = 1;
static int staticGlobalVar = 1;

int main() {
    static int staticVar = 1;
    int localVar = 1;
    int num1[10] = {1, 2, 3, 4};
    char char2[] = "abcd";
    const char *pChar3 = "abcd";
    int *ptr1 = (int *) malloc(sizeof(int) * 4);
    int *ptr2 = (int *) calloc(4, sizeof(int));
    int *ptr3 = (int *) realloc(ptr2, sizeof(int) * 4);
    free(ptr1);
    free(ptr3);
    return 0;
}
//1. 选择题:
// 选项: A.栈 B.堆 C.数据段(静态区) D.代码段(常量区)
// globalVar在哪里?____  staticGlobalVar在哪里?____
// staticVar在哪里?____  localVar在哪里?____
// num1 在哪里?____
//
// char2在哪里?____  *char2在哪里?___
// pChar3在哪里?____   *pChar3在哪里?____
// ptr1在哪里?____ *ptr1在哪里?____
//2. 填空题:
// sizeof(num1) = ____; 
//sizeof(char2) = ____;  strlen(char2) = ____;
// sizeof(pChar3) = ____;   strlen(pChar3) = ____;
// sizeof(ptr1) = ____;
//3. sizeof 和 strlen 区别?
//

  • 说明
    1. 又叫堆栈–非静态局部变量/函数参数/返回值等等,栈是向下增长的。
    2. 内存映射段 是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。(Linux课程如果没学到这块,现在只需要了解一下)
    3. 用于程序运行时动态内存分配,堆是可以上增长的。
    4. 数据段存储全局数据和静态数据
    5. 代码段可执行的代码/只读常量

所以1.选择题有答案了。

那么2.填空题答案是:之前学过不讲了哈哈。

其中3.sizeofstrlen 区别?

答:strlen遇到\0结束,strlen算字符串长度,sizeof算变量大小。


2、C语言中动态内存管理方式:malloc/calloc/realloc/free

malloccallocrealloc 是在C语言中用于动态内存分配的三个函数,它们有一些区别,主要体现在它们的功能和用法上。

  1. malloc(Memory Allocation,内存分配):

    • malloc 是 “memory allocation” 的缩写,用于分配指定大小的内存块。
    • 它只分配内存,不对内存进行初始化,所以分配的内存中可能包含任意值。
    void* malloc(size_t size);
    
  2. calloc(Contiguous Allocation,连续分配):

    • calloc 也是用于分配内存的函数,但与 malloc 不同的是,calloc 分配的内存块会被初始化为零
    • calloc 接受两个参数,分别是所需的元素个数和每个元素的大小。
    void* calloc(size_t num_elements, size_t element_size);
    
  3. realloc(Re-allocation,重新分配):

    • realloc 用于重新分配已分配内存的大小,可以用于扩大或缩小内存块的大小。
    • 如果原始内存块的地址不为空,realloc 会尝试在原始地址上修改内存块的大小(如果新内存大小大于这块原始空间,则还是需要重新开辟内存),如果原始地址为空,则行为类似于 malloc
    void* realloc(void* ptr, size_t new_size);
    

总结:

  • malloc 只分配指定大小的内存块,不进行初始化。
  • calloc 分配指定数量和大小的内存块,并将内存块的所有位初始化为零。
  • realloc 重新分配内存块的大小,可以用于扩大或缩小已分配内存的大小。
int main() {
    int *p1 = (int *) malloc(sizeof(int));
    free(p1);
// 1.malloc/calloc/realloc的区别是什么?
    int *p2 = (int *) calloc(4, sizeof(int));
    int *p3 = (int *) realloc(p2, sizeof(int) * 10);
// 这里需要free(p2)吗? --- 看情况
    free(p3);
    return 0;
}

3、C++动态内存管理

C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力(比如给对象初始化),而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理

3.1、new/delete操作内置类型

int main() {
    // 动态申请一个int类型的空间
    int *ptr4 = new int;

    // 动态申请一个int类型的空间并初始化为10
    int *ptr5 = new int(10);

    cout << *ptr5 << endl;
    // 动态申请10个int类型的空间
    int *ptr6 = new int[3];
    // 动态申请10个int类型的空间并初始化前3个
    int *ptr7 = new int[3]{1, 2, 3};
    cout << ptr7[0] << ptr7[1] << ptr7[2] << ptr7[3];
    
    delete ptr4;
    delete ptr5;
    delete[] ptr6;
    delete[] ptr7;
}
  • 初始化格式
    • 单个元素空间:new 类型 (初始化值)
    • 连续空间:new 类型 [元素个数] {从前往后元素初始化值,其余元素初始化为0}
  • 注意:申请和释放单个元素的空间,使用newdelete操作符,申请和释放连续的空间,使用new[]delete[]。注意:匹配起来使用

3.2、new/delete操作自定义类型

class Date {
public:
    Date() : _year(1), _month(1), _day(1) {
        cout << "Date()" << endl;
    }

    Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
        _year = year;
        _month = month;
        _day = day;
        cout << "Date()" << endl;
    }

    ~Date() {
        cout << "~Date()" << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};


int main() {
    Date *ptr1 = new Date();
    Date *ptr2 = new Date(2, 2, 2);

    Date *ptr3 = new Date[10]{{1, 2, 2}};

    free(ptr1);
    delete ptr2;
    delete[] ptr3;
    return 0;
}

  • 初始化格式
    • 单个元素空间:new 类型 (初始化值)
    • 连续空间:new 类型 [元素个数] {从前往后元素初始化值使用{}代表每一个元素的值,其余元素初始化为0}
  • new/deletemalloc/free最大区别是 new/delete对于【自定义类型】除了开空间还会调用构造函数和析构函数

4、operator new与operator delete函数

newdelete是用户进行动态内存申请和释放的操作符,operator newoperator delete是系统提供的全局函数,new底层调用operator new全局函数来申请空间,delete底层通过operator delete全局函数来释放空间。

  • operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间失败,尝试执行空间不足应对措施,如果改应对措施用户设置了,则继续申请,否则抛异常。
  • operator delete 最终是通过free来释放空间的。
class Date {
public:
    Date() : _year(1), _month(1), _day(1) {
        cout << "Date()" << endl;
    }

    Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
        _year = year;
        _month = month;
        _day = day;
        cout << "Date()" << endl;
    }

    ~Date() {
        cout << "~Date()" << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};


int main() {
    //operator new -- 不调用构造函数 和 malloc 基本一样
    Date *ptr6 = (Date *) operator new(sizeof(Date));

    delete new(ptr6) Date;
    ptr6 = nullptr;
    return 0;
}

5、new和delete的实现原理

5.1、内置类型

如果申请的是内置类型的空间,newmallocdeletefree基本类似,不同的地方是:new/delete申请和释放的是单个元素的空间,new[]delete[]申请的是连续空间,而且new在申请空间失败时会抛异常malloc会返回NULL

5.2、自定义类型

  • new的原理

    1. 调用operator new函数申请空间
    2. 在申请的空间上执行构造函数,完成对象的构造
  • delete的原理

    1. 在空间上执行析构函数,完成对象中资源的清理工作
    2. 调用operator delete函数释放对象的空间

  • new T[N]的原理

    1. 调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请

    2. 在申请的空间上执行N次构造函数

  • delete[]的原理

    1. 在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理

    2. 调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间

这里我们需要注意一个现象对于内置类型在new T[N]时候,往往开辟的内存空间会大于N*sizeof(T),可能会多出4个字节

原本这里ptr2开辟的空间size应该是12*10 = 120,但是这里显示124,其中4个字节是用来存储开辟连续Date对象的个数。

class Date {
public:
    Date() : _year(1), _month(1), _day(1) {
        cout << "Date()" << endl;
    }

    Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
        _year = year;
        _month = month;
        _day = day;
        cout << "Date()" << endl;
    }

    ~Date() {
        cout << "~Date()" << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};


int main() {
    Date* ptr1 = new Date;
    delete ptr1;

    Date* ptr2 = new Date[10];
    delete[] ptr2;
    return 0;
}

但是如果是内置类型new T[N],则不需要另外的空间存个数

原本这里ptr3开辟的空间size应该是4*10 = 40,并且这里显示40,即没有开辟空间存个数。

class Date {
public:
    Date() : _year(1), _month(1), _day(1) {
        cout << "Date()" << endl;
    }

    Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
        _year = year;
        _month = month;
        _day = day;
        cout << "Date()" << endl;
    }

    ~Date() {
        cout << "~Date()" << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};


int main() {
    Date* ptr1 = new Date;
    delete ptr1;

    Date* ptr2 = new Date[10];
    delete[] ptr2;
  
  	int* ptr3 = new int[10];
		delete[] ptr3;
    return 0;
}

原因:这多出来的4个字节是用来记录开辟T大小连续空间的个数,以便于delete [] 进行析构和释放空间。

这里如果不写析构函数(默认成员变量没有开辟空间,如果开辟了空间,必须调用析构函数释放空间,不然会内存泄露)的话就不报错,因为成员变量都是内置类型没有开空间,不需要调用析构函数,也就不需要在前面添加4字节存个数,即自定义类型new T[N]可以直接delete,内置类型也一样。

class Date {
public:
    Date() : _year(1), _month(1), _day(1) {
        cout << "Date()" << endl;
    }

    Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
//        _a = new int[10];
        _year = year;
        _month = month;
        _day = day;
        cout << "Date()" << endl;
    }

//    ~Date() {
//        cout << "~Date()" << endl;
//    }

private:
//    int* _a;
    int _year;
    int _month;
    int _day;
};


int main() {
    Date *ptr1 = new Date;
    delete ptr1;

    Date *ptr2 = new Date[10];
    // 这里如果不写析构函数(默认成员变量没有开辟空间,如果开辟了空间,必须调用析构函数释放空间,不然会内存泄露)的话就不报错
    // 因为成员变量都是内置类型没有开空间,不需要调用析构函数,也就不需要在前面添加4字节存个数
    delete ptr2;
    return 0;
}

原本这里ptr2开辟的空间size应该是12*10 = 120,并且这里显示120,即没有开辟空间存对象个数。


6、定位new表达式(placement-new)

定位new表达式是在已分配的原始内存空间中调用构造函数初始化一个对象

  • 使用格式new (place_address) type或者new (place_address) type(initializer-list)
    • place_address必须是一个指针,initializer-list是类型的初始化列表
  • 使用场景
    • 定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化。
class Date {
public:
    Date() : _year(1), _month(1), _day(1) {
        cout << "Date()" << endl;
    }

    Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
        _year = year;
        _month = month;
        _day = day;
        cout << "Date()" << endl;
    }

    ~Date() {
        cout << "~Date()" << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};


int main() {
    Date *p = (Date *) operator new(sizeof(Date));
    // 不能显式调用构造函数
    // p->Date();
    // 定位new可以显式调用构造函数
    new(p)Date(1, 1, 1);
    p->~Date();

    return 0;
}

OKOK,C/C++内存管理就到这里。如果你对Linux和C++也感兴趣的话,可以看看我的主页哦。下面是我的github主页,里面记录了我的学习代码和leetcode的一些题的题解,有兴趣的可以看看。

Xpccccc的github主页

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

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

相关文章

立刻解决缺少msvcp140_1.dll解决方法,msvcp140_1.dll修复指南

在日常使用电脑的过程中&#xff0c;我们有时会遇到由于某些重要的DLL文件缺失而导致的程序无法正常启动的问题。很多用户可能都经历过由于缺少msvcp140_1.dll导致应用程序无法运行的情况。本文将为你提供解决msvcp140_1.dll缺失问题的详尽方法&#xff0c;附带对每种方法优点和…

景联文科技加入中国人工智能产业联盟(AIIA)数据委员会

近日&#xff0c;景联文科技加入中国人工智能产业联盟&#xff08;AIIA&#xff09;数据委员会&#xff0c;成为委员会成员单位。 中国人工智能产业发展联盟&#xff08;简称AIIA&#xff09;是在国家发改委、科技部、工信部、网信办指导下&#xff0c;由中国信息通信研究院等单…

Kafka 分布式消息系统

文章目录 消息中间件对比Kafka概述kafka安装和配置kafka入门生产者发送消息消费者接收消息 Kafka高可用设计集群备份机制(Replication&#xff09;备份机制(Replication)-同步方式 kafka生产者详解同步发送异步发送参数详解(ack)参数详解(retries)参数详解-消息压缩 kafka消费者…

Day45力扣打卡

打卡记录 无矛盾的最佳球队&#xff08;线性DP&#xff09; class Solution:def bestTeamScore(self, scores: List[int], ages: List[int]) -> int:n len(scores) nums sorted(zip(scores, ages))f sorted(scores)for i in range(n):for j in range(0, i):if nu…

P8A012-A016组策略安全

账户策略 【预备知识】 组策略&#xff08;Group Policy&#xff09;是Microsoft Windows系统管理员为用户和计算机定义并控制程序、网络资源及操作系统行为的主要工具。通过使用组策略可以设置各种软件、计算机和用户策略。 【实验步骤】 网络拓扑&#xff1a;server2008A…

【开发实践】使用jstree实现文件结构目录树

一、需求分析 因开发系统的需要&#xff0c;维护服务端导出文件的目录结构。因此&#xff0c;需要利用jstree&#xff0c;实现前端对文件结构目录的展示。 【预期效果】&#xff1a; 二、需求实现 【项目准备】&#xff1a; jstree在线文档&#xff1a;jstree在线文档地址 …

(C++)移动零--双指针法

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台备战技术面试&#xff1f;力扣提供海量技术面试资源&#xff0c;帮助你高效提升编程技能&#xff0c;轻松拿下世界 IT 名企 Dream Offer。https://le…

ZooKeeper的分布式锁---客户端命令行测试(实操课程)

本系列是zookeeper相关的实操课程&#xff0c;课程测试环环相扣&#xff0c;请按照顺序阅读测试来学习zookeeper。阅读本文之前&#xff0c;请先阅读----​​​​​​zookeeper 单机伪集群搭建简单记录&#xff08;实操课程系列&#xff09;。 阅读本文之前&#xff0c;请先阅读…

【Vue3】源码解析-虚拟DOM

【Vue3】源码解析 系列文章什么是虚拟DOMVue 3虚拟DOM获取<template>内容生成AST语法树生成render方法字符串得到最终VNode对象 系列文章 【Vue3】源码解析-前置 【Vue3】源码解析-响应式原理 【Vue3】源码解析-虚拟DOM 什么是虚拟DOM 在浏览器中&#xff0c;HTML页面…

基于asp.net 消防安全宣传网站设计与实现

目 录 1 绪论 1 1.&#xff11;课题背景 1 1.2 目的和意义 1 1.3主要研究内容 1 1.4 组织结构 2 2 可行性分析 3 2.1技术可行性 3 2.2经济可行性 3 2.3操作可行性 3 2.4系统开发环境 4 3 需求分析 7 3.1性能分析 7 3.2业务流程分析 7 3.3数据流程分析 9 4 系统设计 11 4.1系统…

创建SpringBoot流程

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、操作步骤总结 前言 我使用的是jdk1.8版本的&#xff0c;idea社区版的&#xff08;专业版的应该也差不多吧&#xff09; 提示…

Leetcode算法之哈希表

目录 1.两数之和2.判定是否互为字符重排3.存在重复元素I4.存在重复元素II5.字母异位词分组 1.两数之和 两数之和 class Solution { public:vector<int> twoSum(vector<int>& nums, int target) {unordered_map<int,int> hash;for(int i0;i<nums.si…

数据结构---二叉树

二叉树的概念及结构 1.概念 一棵二叉树是结点的一个有限集合&#xff0c;该集合: 或者为空 由一个根节点加上两棵别称为左子树和右子树的二叉树组成 从上图可以看出&#xff1a; 二叉树不存在度大于2的结点 二叉树的子树有左右之分&#xff0c;次序不能颠倒&#xff0c;因此…

某60物联网安全之IoT漏洞利用实操2学习记录

物联网安全 文章目录 物联网安全IoT漏洞利用实操2&#xff08;内存破坏漏洞&#xff09;实验目的实验环境实验工具实验原理实验内容实验步骤ARM ROP构造与调试MIPS栈溢出漏洞逆向分析 IoT漏洞利用实操2&#xff08;内存破坏漏洞&#xff09; 实验目的 学会ARM栈溢出漏洞的原理…

Unity C++交互

一、设置Dll输出。 两种方式&#xff1a; 第一&#xff1a;直接创建动态链接库工程第二&#xff1a;创建的是可执行程序&#xff0c;在visual studio&#xff0c;右键项目->属性(由exe改成dll) 二、生成Dll 根据选项Release或Debug&#xff0c;运行完上面的生成解决方案后…

FPGA设计时序约束十、others类约束之Set_Disable_Timing

目录 一、序言 二、Set Disable Timing 2.1 基本概念 2.2 设置界面 2.3 命令语法 2.4 命令示例 三、工程示例 四、参考资料 一、序言 在Vivado的时序约束窗口中&#xff0c;存在一类特殊的约束&#xff0c;划分在others目录下&#xff0c;可用于设置忽略或修改默认的时…

7.浮点数转为整数【2023.11.29】

1.问题描述 给出一个浮点数&#xff0c;请将这个浮点数转换成整数。 2.解决思路 输入一个浮点数。 输出程序将浮点数转换为整数并输出。 3.代码实现 numfloat(input("请输入一个浮点数")) num1int(num) print(num1)4.运行结果

智能优化算法应用:基于萤火虫算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于萤火虫算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于萤火虫算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.萤火虫算法4.实验参数设定5.算法结果6.参考文献7.…

机关单位档案分类及整理方法

机关单位档案主要包含文书档案、干部职工档案&#xff08;人事档案&#xff09;、会计档案、科技档案&#xff08;科学研究、基本建设、设备仪器、产品&#xff09;、诉讼档案、音像档案、照片档案、电子档案等等&#xff0c;这其中&#xff0c;不同种类&#xff0c;不同载体的…

技术前沿丨Teranode如何实现无限扩容

​​发表时间&#xff1a;2023年9月15日 BSV区块链协会的技术团队目前正在努力开发Teranode&#xff0c;这是一款比特币节点软件&#xff0c;其最终目标是实现比特币的无限扩容。然而&#xff0c;正如BSV区块链协会网络基础设施负责人Jake Jones在2023年6月举行的伦敦区块链大会…