• C语言 内存


    内存分配

    内存分配的类型

    C/C++中内存分为5个区,分别为栈区、堆区、全局/静态存储区、常量存储区、代码区

    静态内存分配:编译时分配,包括全局、静态全局、静态局部三种变量。

    动态内存分配:运行时分配,包括栈(局部变量),堆(c语言中用到的变量被动态的分配在内存中)。

    变量的内存分配

    栈区

    介绍
    • 栈区由编译器自动分配释放,由操作系统自动管理,无须手动管理。
    • 栈区上的内容只在函数范围内存在,当函数运行结束,这些内容也会自动被销毁。
    • 栈区按内存地址由高到低方向生长,其最大大小由编译时确定,速度快,但自由性差,最大空间不大。
    • 栈区是先进后出原则,即先进去的被堵在屋里的最里面,后进去的在门口,释放的时候门口的先出去。
    存放内容
    • 临时创建的局部变量和const定义的局部变量存放在栈区。
    • 函数调用和返回时,其入口参数和返回值存放在栈区。

    堆区

    介绍
    • 堆区由程序员分配内存和释放。
    • 堆区按内存地址由低到高方向生长,其大小由系统内存/虚拟内存上限决定,速度较慢,但自由性大,可用空间大。

    全局/静态区

    • 通常是用于那些在编译期间就能确定存储大小的变量的存储区,但它用于的是在整个程序运行期间都可见的全局变量和静态变量。
    • 全局区有 .bss段 和 .data段组成,可读可写。
    .bss段
    • 未初始化的全局变量和未初始化的静态变量存放在 .bss 段。
    • 初始化为 0 的全局变量和初始化为0的静态变量存放在 .bss 段。
    • .bss 段不占用可执行文件空间,其内容由操作系统初始化。
    .data段
    • 已初始化的全局变量存放在 .data 段。
    • 已初始化的静态变量存放在 .data 段。
    • .data 段占用可执行文件空间,其内容有程序初始化。

    常量区

    • 字符串、数字等常量存放在常量区。
    • const修饰的全局变量存放在常量区。
    • 程序运行期间,常量区的内容不可以被修改。

    代码区

    • 程序执行代码存放在代码区,其值不能修改(若修改则会出现错误)。
    • 字符串常量和define定义的常量也有可能存放在代码区。

    栈区和堆区的比较

    申请方式

    栈由编译器管理,堆的分配和释放由程序员管理。

    申请大小

    栈是向低地址生长的数据结构,是一块连续的内存,能从栈中获得的内存较小,编译期间确定大小;堆是向高地址生长的数据结构,是一个不连续的储存空间,内存获取比较灵活,也较大。

    栈与堆中的储存内容

    栈:在函数调用时,第一个进栈的是主函数中的最后一条指令的地址,然后是函数的各个参数,在大多 数的c编译器中,参数是由右往左入栈的,然后是函数中的局部变量(静态变量是不入栈的),当本次函数调用结束后,局部变量先出栈,然后是参数,最后栈顶指针指向最开始存的地址,也就是主函数中的下一条指令,程序由该点开始运行;

    堆:一般是在堆的头部用一个字节存放堆的大小,堆中的具体内容由程序员安排。

    动态内存分配

    malloc 函数

    头文件

    stdlib.hmalloc.h

    函数原型

    void* malloc(int size)

    功能

    分配长度为 size 字节的内存块

    参数

    只有一个参数,并且是整型

    返回值

    分配成功返回指向被分配内存的指针,否则返回空指针 NULL

    返回值类型 void* ,在 C 语言中,void* 可以不经转换直接赋值给任何类型的指针变量(函数指针变量除外)

    calloc 函数

    头文件

    stdlib.hmalloc.h

    函数原型

    void* calloc(int n,int size)

    功能

    分配 n 个长度为 size 字节的连续空间

    返回值

    分配成功返回指向分配起始地址的指针,否则返回空指针 NULL

    realloc 函数

    头文件

    stdlib.hmalloc.h

    函数原型

    void* realloc(void* mem_address,int newsize)

    功能

    1.为已有内存的变量重新分配新的内存空间(可大、可小) ;

    2.先判断当前的指针是否有足够的连续空间,如果有,扩大 mem_address 指向的地址,并且将 mem_ address 返回;

    3.如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来 mem_address 所指内存区域(注意:原来指针是自动释放,不需要使用 free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。

    返回值

    分配成功返回 mem_address 所指的内存区域,否则返回空指针 NULL

    free 函数

    动态分配的内存空间由程序员手动编程释放。

    C 语言中,free 和 malloc、calloc、realloc 是成对出现的,有多少个 malloc,就应该有多少个 free 。

    若 malloc、calloc、realloc 的个数比 free 多,会造成内存泄漏;若 malloc、calloc、realloc 的个数比 free 少,会造成二次删除,破坏内存,导致程序崩溃。

    头文件

    stdlib.h

    函数声明

    void free(void* ptr)

    功能

    释放之前调用 calloc、malloc 或 realloc 所分配的内存空间

    参数

    指针指向一个要释放内存的内存块,该内存块之前是通过调用 malloc、calloc 或 realloc 进行分配内存的。如果传递的参数是一个空指针,则不会执行任何动作。

    返回值

    不返回任何值

  • 相关阅读:
    小波去噪算法的简易实现及其扩展(小波锐化、高斯拉普拉斯金字塔去噪及锐化)之二。
    yarn的安装及使用
    opensd开源啦 ,这套自动化部署OpenStack工具你值得拥有
    【AI学习】Transformer的Token嵌入表示为什么那么长
    Nacos Discovery--服务治理
    【学习笔记】模拟
    IDEA2020.3无法输入中文问题解决
    vue封装wangEditor
    MySQL从基础到毕业【完整篇】
    Node.js安装使用
  • 原文地址:https://blog.csdn.net/weixin_65951505/article/details/133871024