c语言----------内存管理

内存管理

目录

  • 一。作用域
    • 1.1 局部变量
    • 1.2 静态(static)局部变量
    • 1.3 全局变量
    • 1.4 静态(static)全局变量
    • 1.5 extern全局变量声明
    • 1.6 全局函数和静态函数
    • 1.7 总结
  • 二。内存布局
    • 2.1 内存分区
    • 2.2 存储类型总结
    • 2.3内存操作函数
      • 1) memset()
      • 2) memcpy()
      • 3) memmove()
      • 4) memcmp()
    • 2.4 堆区内存分配和释放
      • 1)malloc()
      • 2)free()
  • 3 内存分区代码分析
    • 1) 返回栈区地址
    • 2) 返回data区地址
    • 3) 值传递1
    • 4) 值传递2
    • 5) 返回堆区地址

一。作用域

C语言变量的作用域分为:

  • 代码块作用域(代码块是{}之间的一段代码)
  • 函数作用域
  • 文件作用域

1.1 局部变量

局部变量也叫auto自动变量(auto可写可不写),一般情况下代码块{}内部定义的变量都是自动变量,它有如下特点:

  • 在一个函数内定义,只在函数范围内有效
  • 在复合语句中定义,只在复合语句中有效
  • 随着函数调用的结束或复合语句的结束局部变量的声明声明周期也结束
  • 如果没有赋初值,内容为随机
#include <stdio.h>void test()
{//auto写不写是一样的//auto只能出现在{}内部auto int b = 10; 
}int main(void)
{//b = 100; //err, 在main作用域中没有bif (1){//在复合语句中定义,只在复合语句中有效int a = 10;printf("a = %d\n", a);}//a = 10; //err离开if()的复合语句,a已经不存在return 0;
}

1.2 静态(static)局部变量

  • static局部变量的作用域也是在定义的函数内有效
  • static局部变量的生命周期和程序运行周期一样,同时staitc局部变量的值只初始化一次,但可以赋值多次
  • static局部变量若未赋以初值,则由系统自动赋值:数值型变量自动赋初值0,字符型变量赋空字符
#include <stdio.h>void fun1()
{int i = 0;i++;printf("i = %d\n", i);
}void fun2()
{//静态局部变量,没有赋值,系统赋值为0,而且只会初始化一次static int a;a++;printf("a = %d\n", a);
}int main(void)
{fun1();fun1();fun2();fun2();return 0;
}

1.3 全局变量

  • 在函数外定义,可被本文件及其它文件中的函数所共用,若其它文件中的函数调用此变量,须用extern声明
  • 全局变量的生命周期和程序运行周期一样

1.4 静态(static)全局变量

  • 在函数外定义,作用范围被限制在所定义的文件中
  • 不同文件静态全局变量可以重名,但作用域不冲突
  • static全局变量的生命周期和程序运行周期一样,同时staitc全局变量的值只初始化一次

1.5 extern全局变量声明

extern int a;声明一个变量,这个全局变量在别的文件中已经定义了,这里只是声明,而不是定义。

1.6 全局函数和静态函数

在C语言中函数默认都是全局的,使用关键字static可以将函数声明为静态,函数定义为static就意味着这个函数只能在定义这个函数的文件中使用,在其他文件中不能调用,即使在其他文件中声明这个函数都没用。

对于不同文件中的staitc函数名字可以相同。

注意

  • 允许在不同的函数中使用相同的变量名,它们代表不同的对象,分配不同的单元,互不干扰。
  • 同一源文件中,允许全局变量和局部变量同名,在局部变量的作用域内,全局变量不起作用。
  • 所有的函数默认都是全局的,意味着所有的函数都不能重名,但如果是staitc函数,那么作用域是文件级的,所以不同的文件static函数名是可以相同的。

1.7 总结

类型	      			作用域				生命周期
auto变量				一对{}内		当前函数
static局部变量			一对{}内		整个程序运行期
extern变量				整个程序		整个程序运行期
static全局变量			当前文件		整个程序运行期
extern函数				整个程序		整个程序运行期
static函数				当前文件		整个程序运行期
register变量			一对{}内			当前函数
全局变量				整个程序		 整个程序运行期

二。内存布局

2.1 内存分区

C代码经过预处理、编译、汇编、链接4步后生成一个可执行程序。
在 Windows 下,程序是一个普通的可执行文件,以下列出一个二进制可执行文件的基本情况:
在这里插入图片描述

通过上图可以得知,在没有运行程序前,也就是说程序没有加载到内存前,可执行程序内部已经分好3段信息,分别为代码区(text)、数据区(data)和未初始化数据区(bss)3 个部分(有些人直接把data和bss合起来叫做静态区或全局区)。

代码区
存放 CPU 执行的机器指令。通常代码区是可共享的(即另外的执行程序可以调用它),使其可共享的目的是对于频繁被执行的程序,只需要在内存中有一份代码即可。代码区通常是只读的,使其只读的原因是防止程序意外地修改了它的指令。另外,代码区还规划了局部变量的相关信息。

全局初始化数据区/静态数据区(data段)
该区包含了在程序中明确被初始化的全局变量、已经初始化的静态变量(包括全局静态变量和局部静态变量)和常量数据(如字符串常量)。

未初始化数据区(又叫 bss 区)
存入的是全局未初始化变量和未初始化静态变量。未初始化数据区的数据在程序开始执行之前被内核初始化为 0 或者空(NULL)。

程序在加载到内存前,代码区和全局区(data和bss)的大小就是固定的,程序运行期间不能改变。然后,运行可执行程序,系统把程序加载到内存,除了根据可执行程序的信息分出代码区(text)、数据区(data)和未初始化数据区(bss)之外,还额外增加了栈区、堆区。

在这里插入图片描述

  • 栈区(stack)
    栈是一种先进后出的内存结构,由编译器自动分配释放,存放函数的参数值、返回值、局部变量等。在程序运行过程中实时加载和释放,因此,局部变量的生存周期为申请到释放该段栈空间。

  • 堆区(heap)
    堆是一个大容器,它的容量要远远大于栈,但没有栈那样先进后出的顺序。用于动态内存分配。堆在内存中位于BSS区和栈区之间。一般由程序员分配和释放,若程序员不释放,程序结束时由操作系统回收。

  • 未初始化数据区(BSS)
    加载的是可执行文件BSS段,位置可以分开亦可以紧靠数据段,存储于数据段的数据(全局未初始化,静态未初始化数据)的生存周期为整个程序运行过程。

  • 全局初始化数据区/静态数据区(data segment)
    加载的是可执行文件数据段,存储于数据段(全局初始化,静态初始化数据,文字常量(只读))的数据的生存周期为整个程序运行过程。

  • 代码区(text segment)
    加载的是可执行文件代码段,所有的可执行代码都加载到代码区,这块内存是不可以在运行期间修改的。

2.2 存储类型总结

类型			作用域			生命周期			存储位置
auto变量		一对{}内	当前函数				栈区
static局部变量	一对{}内	整个程序运行期			初始化在data段,未初始化在BSS段
extern变量		整个程序	整个程序运行期			初始化在data段,未初始化在BSS段
static全局变量	当前文件	整个程序运行期			初始化在data段,未初始化在BSS段
extern函数		整个程序	整个程序运行期			代码区
static函数		当前文件	整个程序运行期			代码区
register变量	一对{}内	当前函数				运行时存储在CPU寄存器
字符串常量		当前文件	整个程序运行期			data段

代码实例

	#include <stdio.h>#include <stdlib.h>int e;static int f;int g = 10;static int h = 10;int main(){printf("--栈--\n");printf("--堆--\n");printf("--BSS--\n");printf("--data--\n");printf("--text--\n");int a;int b = 10;char* k = NULL;static int c;static int d = 10;char* i = "test";printf("&a\t %p\t //局部未初始化变量---栈\n", &a);printf("&b\t %p\t //局部初始化变量---栈\n", &b);k = (char*)malloc(10);printf("k\t %p\t //动态分配的内存---堆\n", k);printf("&e\t %p\t //全局未初始化变量---BSS\n", &e);printf("&c\t %p\t //静态局部未初始化变量--BSS\n", &c);printf("&f\t %p\t //全局静态未初始化变量---BSS\n", &f);printf("&g\t %p\t //全局初始化变量---data\n", &g);printf("&d\t %p\t //静态局部初始化变量---data\n", &d);printf("&h\t %p\t //全局静态初始化变量---data\n", &h);printf("i\t %p\t //只读数据(文字常量区)---data\n", i);return 0;}

2.3内存操作函数

1) memset()

#include <string.h>
void *memset(void *s, int c, size_t n);
功能:将s的内存区域的前n个字节以参数c填入
参数:s:需要操作内存s的首地址c:填充的字符,c虽然参数为int,但必须是unsigned char , 范围为0~255n:指定需要设置的大小
返回值:s的首地址
int a[10];printf("-----------\n");memset(a, 0, sizeof(a));for (int i = 0; i < 10; i++){printf("%c\n", a[i]);}printf("-----------\n");memset(a, 97, sizeof(a));for (int i = 0; i < 10; i++){printf("%c\n", a[i]);}

2) memcpy()

#include <string.h>
void *memcpy(void *dest, const void *src, size_t n);
功能:拷贝src所指的内存内容的前n个字节到dest所值的内存地址上。
参数:dest:目的内存首地址src:源内存首地址,注意:dest和src所指的内存空间不可重叠,可能会导致程序报错n:需要拷贝的字节数
返回值:dest的首地址
int a[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
int b[10];memcpy(b, a, sizeof(a));
int i = 0;
for (i = 0; i < 10; i++)
{printf("%d, ", b[i]);
}
printf("\n");//memcpy(&a[3], a, 5 * sizeof(int)); //err, 内存重叠

3) memmove()

memmove()功能用法和memcpy()一样,区别在于:dest和src所指的内存空间重叠时,memmove()仍然能处理,不过执行效率比memcpy()低些。

4) memcmp()

#include <string.h>
int memcmp(const void *s1, const void *s2, size_t n);
功能:比较s1和s2所指向内存区域的前n个字节
参数:s1:内存首地址1s2:内存首地址2n:需比较的前n个字节
返回值:相等:=0大于:>0小于:<0
int a[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
int b[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int flag = memcmp(a, b, sizeof(a));
printf("flag = %d\n", flag);

2.4 堆区内存分配和释放

1)malloc()

#include <stdlib.h>
void *malloc(size_t size);
功能:在内存的动态存储区(堆区)中分配一块长度为size字节的连续区域,用来存放类型说明符指定的类型。分配的内存空间内容不确定,一般使用memset初始化。
参数:size:需要分配内存大小(单位:字节)
返回值:
成功:分配空间的起始地址
失败:NULL
#include <stdlib.h> 
#include <stdio.h>
#include <string.h>int main()
{int count, *array, n;printf("请输入要申请数组的个数:\n");scanf("%d", &n);array = (int *)malloc(n * sizeof (int));if (array == NULL){printf("申请空间失败!\n");return -1;}//将申请到空间清0memset(array, 0, sizeof(int)*n);for (count = 0; count < n; count++) /*给数组赋值*/array[count] = count;for (count = 0; count < n; count++) /*打印数组元素*/printf("%2d", array[count]);free(array);return 0;
}

2)free()

#include <stdlib.h>
void free(void *ptr);
功能:释放ptr所指向的一块内存空间,ptr是一个任意类型的指针变量,指向被释放区域的首地址。对同一内存空间多次释放会出错。
参数:
ptr:需要释放空间的首地址,被释放区应是由malloc函数所分配的区域。
返回值:无

3 内存分区代码分析

1) 返回栈区地址

#include <stdio.h>
int *fun()
{int a = 10;return &a;//函数调用完毕,a释放
}int main(int argc, char *argv[])
{int *p = NULL;p = fun();*p = 100; //操作野指针指向的内存,errreturn 0;
}

2) 返回data区地址

#include <stdio.h>int *fun()
{static int a = 10;return &a; //函数调用完毕,a不释放
}int main(int argc, char *argv[])
{int *p = NULL;p = fun();*p = 100; //okprintf("*p = %d\n", *p);return 0;
}

3) 值传递1

#include <stdio.h>
#include <stdlib.h>void fun(int *tmp)
{tmp = (int *)malloc(sizeof(int));*tmp = 100;
}int main(int argc, char *argv[])
{int *p = NULL;fun(p); //值传递,形参修改不会影响实参printf("*p = %d\n", *p);//err,操作空指针指向的内存return 0;
}

4) 值传递2

#include <stdio.h>
#include <stdlib.h>void fun(int *tmp)
{*tmp = 100;
}int main(int argc, char *argv[])
{int *p = NULL;p = (int *)malloc(sizeof(int));fun(p); //值传递printf("*p = %d\n", *p); //ok,*p为100return 0;
}

5) 返回堆区地址

#include <stdio.h>
#include <stdlib.h>int *fun()
{int *tmp = NULL;tmp = (int *)malloc(sizeof(int));*tmp = 100;return tmp;//返回堆区地址,函数调用完毕,不释放
}int main(int argc, char *argv[])
{int *p = NULL;p = fun();printf("*p = %d\n", *p);//ok//堆区空间,使用完毕,手动释放if (p != NULL){free(p);p = NULL;}return 0;
}

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

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

相关文章

【2024年华为OD机试】 (C卷,100分)- 堆栈中的剩余数字(Java JS PythonC/C++)

一、问题描述 题目描述 向一个空栈中依次存入正整数&#xff0c;假设入栈元素 n(1<n<2^31-1)按顺序依次为 nx…n4、 n3、n2、 n1, 每当元素入栈时&#xff0c;如果 n1n2…ny(y 的范围[2,x]&#xff0c; 1<x<1000)&#xff0c;则 n1~ny 全部元素出栈&#xff0c;重…

Java安全—SPEL表达式XXESSTI模板注入JDBCMyBatis注入

前言 之前我们讲过SpringBoot中的MyBatis注入和模板注入的原理&#xff0c;那么今天我们就讲一下利用以及发现。 这里推荐两个专门研究java漏洞的靶场&#xff0c;本次也是根据这两个靶场来分析代码&#xff0c;两个靶场都是差不多的。 https://github.com/bewhale/JavaSec …

51单片机入门基础

目录 一、基础知识储备 &#xff08;一&#xff09;了解51单片机的基本概念 &#xff08;二&#xff09;掌握数字电路基础 &#xff08;三&#xff09;学习C语言编程基础 二、开发环境搭建 &#xff08;一&#xff09;硬件准备 &#xff08;二&#xff09;软件准备 三、…

基于Java的百度AOI数据解析与转换的实现方法

目录 前言 一、AOI数据结构简介 1、官网的实例接口 2、响应参数介绍 二、Java对AOI数据的解析 1、数据解析流程图 2、数据解析实现 3、AOI数据解析成果 三、总结 前言 在当今信息化社会&#xff0c;地理信息数据在城市规划、交通管理、商业选址等领域扮演着越来越重要的…

【WEB】网络传输中的信息安全 - 加密、签名、数字证书与HTTPS

文章目录 1. 概述2. 网络传输安全2.1.什么是中间人攻击2.2. 加密和签名2.2.1.加密算法2.2.2.摘要2.2.3.签名 2.3.数字证书2.3.1.证书的使用2.3.2.根证书2.3.3.证书链 2.4.HTTPS 1. 概述 本篇主要是讲解讲一些安全相关的基本知识&#xff08;如加密、签名、证书等&#xff09;&…

shell练习2

需求&#xff1a;判断192.168.1.0/24网络中&#xff0c;当前在线的ip有哪些&#xff0c;并编写脚本打印出来。 #!/bin/bashnmap -sn 192.168.1.0/24 | grep Nmap scan report for | awk {print $5} 注意&#xff1a;当运行 bash ip.sh 时出现 nmap: command not found 的错误…

【运维自动化-作业平台】魔法变量到底如何使用之主机列表类型

蓝鲸作业平台&#xff0c;以下简称作业平台或JOB平台 魔法变量&#xff1a;JOB平台执行引擎提供的特有的变量能力用法 脚本中使用&#xff0c;并且需要事先声明&#xff1a;job_import {{变量名}} 声明后&#xff0c;同样是使用 dollar 符 大括号&#xff1a;${变量名}来取值…

活动预告 | CCF开源发展委员会开源供应链安全技术研讨会(2025第一期)——“大模型时代的开源供应链安全风控技术”...

点击蓝字 关注我们 CCF Opensource Development Committee CCF开源发展委员会开源供应链安全工作组&#xff08;以下简称CCF-ODC-OSS&#xff09;将于1月17日下午在北京黄大年茶思屋举行2025年第一期开源供应链安全技术研讨会&#xff0c;此次研讨会主题为“大模型时代的开源供…

XML序列化和反序列化的学习

1、基本介绍 在工作中&#xff0c;经常为了调通上游接口&#xff0c;从而对请求第三方的参数进行XML序列化&#xff0c;这里常使用的方式就是使用JAVA扩展包中的相关注解和类来实现xml的序列化和反序列化。 2、自定义工具类 import javax.xml.bind.JAXBContext; import javax.x…

基于php求职招聘系统设计

基于php求职招聘系统设计 摘要 随着社会信息化时代的到来&#xff0c;如今人们社会的生活节奏普遍加快&#xff0c;人们对于工作效率的要求也越来越高&#xff0c;企业 举办招聘会耗时耗财&#xff0c;个人参加招聘会漫无目的寻找不到“方向”&#xff0c;网络搜索工作量目的…

SDK调用文心一言如何接入,文心一言API接入教程

一、前期准备 注册百度智能云账号&#xff1a; 前往百度智能云官网注册一个账号。这是接入文心一言API的基础。 了解API接口&#xff1a; 在百度智能云开放平台中&#xff0c;找到文心一言API的详情页&#xff0c;了解提供的API接口类型&#xff08;如云端API、移动端API、离线…

【机器学习】数据拟合-最小二乘法(Least Squares Method)

最小二乘法&#xff08;Least Squares Method&#xff09; 最小二乘法是一种广泛使用的数据拟合方法&#xff0c;用于在统计学和数学中找到最佳拟合曲线或模型&#xff0c;使得观测数据点与模型预测值之间的误差平方和最小化。以下是详细介绍&#xff1a; 基本概念 假设有一组…

Flutter 多终端测试 自定义启动画面​​​​​​​ 更换小图标和应用名称

多终端测试 flutter devices flutter run -d emulator-5554 flutter run -d emulator-5556 自定义启动画面 之前&#xff1a; 进入assert 3x 生成 1x 2x dart run flutter_native_splash:create dart run flutter_native_splash:remove 现在&#xff08;flutter_nativ…

springMVC实现文件上传

目录 一、创建项目 二、引入依赖 三、web.xml 四、编写上传文件的jsp页面 五、spring-mvc.xml 六、controller 七、运行 一、创建项目 二、引入依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.o…

Java内存与缓存

Java内存管理和缓存机制是构建高性能应用程序的关键要素。它们之间既有联系又有区别&#xff0c;理解这两者对于优化Java应用至关重要。 Java 内存模型 Java内存模型&#xff08;JMM&#xff09;定义了线程如何以及何时可以看到其他线程修改过的共享变量的值&#xff0c;并且规…

图片和短信验证码(头条项目-06)

1 图形验证码接口设计 将后端⽣成的图⽚验证码存储在redis数据库2号库。 结构&#xff1a; {img_uuid:0594} 1.1 创建验证码⼦应⽤ $ cd apps $ python ../../manage.py startapp verifications # 注册新应⽤ INSTALLED_APPS [django.contrib.admin,django.contrib.auth,…

云服务信息安全管理体系认证,守护云端安全

在数据驱动的时代&#xff0c;云计算已成为企业业务的超级引擎&#xff0c;推动着企业飞速发展。然而&#xff0c;随着云计算的广泛应用&#xff0c;信息安全问题也日益凸显&#xff0c;如同暗流涌动下的礁石&#xff0c;时刻威胁着企业的航行安全。这时&#xff0c;云服务信息…

LabVIEW与WPS文件格式的兼容性

LabVIEW 本身并不原生支持将文件直接保存为 WPS 格式&#xff08;如 WPS 文档或表格&#xff09;。然而&#xff0c;可以通过几种间接的方式实现这一目标&#xff0c;确保您能将 LabVIEW 中的数据或报告转换为 WPS 可兼容的格式。以下是几种常见的解决方案&#xff1a; ​ 导出…

创建 WordPress 插件(第一部分):添加管理页面

WordPress 是互联网上最受欢迎的内容管理系统之一。它是用 PHP 创建的&#xff0c;可以处理从博客到商业网站的一切需求。事实上&#xff0c;我们的博客和网站都使用 WordPress。在本文中&#xff0c;我将向你展示如何创建一个 WordPress 插件&#xff0c;该插件会在管理员控制…

解锁企业数据管理统一身份认证难题,EasyMR助力企业敏捷提效

在数字经济迅猛发展的当下&#xff0c;企业数据量正以令人惊叹的速度持续增长。据IDC研究显示&#xff0c;至2025年&#xff0c;全球数据总量预计将超175 ZB。数据的爆发式增长对企业而言&#xff0c;既是机遇&#xff0c;更是巨大挑战。 如今&#xff0c;大数据已然成为企业决…