C语言进阶 ~ 内存四区(栈、堆、全局、代码区)

2023-05-16

特别声明:该部分是根据B站大佬---什么都想干好的视频学习而来。

目录

1.1 数据类型本质分析

1.1.1 数据类型概念

1.1.2 数据类型的本质

1.1.3 数据类型的别名

1.1.4 数据类型之 void

1.2 变量的本质分析

1.2.1 变量的概念

1.3 程序的内存四区模型

1.3.1 全局区(全局变量、静态变量(const,constant或final等)、文字常量区)

1.3.2 栈区(栈区(stack) :① 由编译器自动分配释放,存放函数的参数值,局部变量的值等。② 函数运行时分配,函数结束时释放。由编译器自动分配释放 ,存放为运行函数而分配的局部变量、函数参数、返回数据、返回地址等。)

1.3.3 堆区(heap) : 一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能由操作系统回收。

1.4 函数的调用模型

1.5 栈的生长方向和内存存放方向


1.1 数据类型本质分析

1.1.1 数据类型概念

  • “类型”是对数据的抽象
  •  类型相同的数据有相同的表示形式、存储格式以及相关的操作
  •  程序中使用的所有数据都必定属于某一种数据类型

1.1.2 数据类型的本质

  • 数据类型可理解为创建变量的模具:是固定内存大小的别名
  • 数据类型的作用:编译器预算对象(变量)分配的内存空间大小。
  •  注意:数据类型只是模具,编译器并没有分配空间,只有根据类型(模具)创建变量(实物),编译器才会分配空间。
#include <stdio.h>

int main(void)
{
    int a = 10; //告诉编译器,分配4个字节的内存
    int b[10];  //告诉编译器,分配4*10 = 40 个字节的内存

    printf("b:%p, b+1: %p, &b:%p, &b+1: %p\n", b, b + 1, &b, &b + 1);

    //b+1 和 &b+1的结果不一样 (+1 ---> +4; +1 ---> +40)
    //是因为 b 和 &b 所代表的数据类型不一样
    //b  代表数组首元素的地址
    //&b 代表整体数组的地址

    return 0;
}
  • b+1 和 &b+1的结果不一样 (+1 ---> +4; +1 ---> +40)
  • 是因为 b 和 &b 所代表的数据类型不一样
  • b  代表数组首元素的地址
  • &b 代表整体数组的地址

1.1.3 数据类型的别名

① 给数据类型起别名

#include <stdio.h>

typedef unsigned int u32;   //给unsigned int类型取别名

int main(void)

{

    u32 a;         

    a = 10;                 

    return 0;                   
}

② 给结构体类型起别名

#include <stdio.h>

#define pi 3.14

// 正常使用结构体 //
struct People
{
	char name[64];
	int age;
};
 给结构体类型起别名 
typedef struct People_2
{
	char name[64];
	int age;
} people_t;


int main(void)
{
	// 正常使用结构体 ///的初始化///
	struct People p1;
	 给结构体类型起别名 /的初始化///
	people_t p2;

	p1.age = 10;
	p2.age = 11;


	return 0;
}

1.1.4 数据类型之 void

1、函数参数为空,定义函数时,可以用void修饰: int fun(void)
2、函数没有返回值: void fun(void);
3、不能定义vold类型的普通变量, vold a; //err,无法确定类型,不同类型分配空间不一样
4、 可以定义vold *变量: void *; //ok, 32位是4字节,64位是8字节
5、数据类型本质:固定内存块大小别名
6、void和万能指针,函数返回值,函数参数

  • void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据。
  • void * memcpy(void *dest, const void *src, size_t len);void指针的意义

7、void指针的意义

      C语言规定只有相同类型的指针才可以相互赋值

      void*指针作为左值用于“接收”任意类型的指针

      void*指针作为右值赋值给其它指针时需要强制类型转换

      int *p1 = NULL;

      char *p2 = (char *)malloc(sizoeof(char)*20);

1.2 变量的本质分析

1.2.1 变量的概念

       概念:既能读又能写的内存对象,称为变量。

#include <stdio.h>
#include <stdlib.h>

int main(void)
{
	int i = 0;

	// 通过变量直接操作内存
	i = 10;

	int *p = &i;
	printf("&i:%d\n", &i);
	printf("p:%d\n", p);

	// 通过内存编号间接操作内存
	*p = 100;
	printf("i = %d, *p = %d\n", i, *p);

	system("pause");
 	return 0;
}

1.3 程序的内存四区模型

1.3.1 全局区(全局变量、静态变量(const,constant或final等)、文字常量区)

#include <stdio.h>

char * getStr1()
{
    char *p1 = "abcdefg2";
    return p1;
}
char *getStr2()
{
    char *p2 = "abcdefg2";
    return p2;
}
int main(void)
{
    char *p1 = NULL;
    char *p2 = NULL;
    p1 = getStr1();
    p2 = getStr2();

    //打印p1 p2 所指向内存空间的数据
    printf("p1:%s , p2:%s \n", p1, p2);

    //打印p1 p2 的值
    printf("p1:%p , p2:%p \n", p1, p2);

    return 0;
}

问题:内容一致, 为什么两个指针的地址值也是一样的?
 

① 程序执行到   int main(void) 

② 程序执行到       char *p1 = NULL;  char *p2 = NULL;

③ 程序执行到 getStr1()        由于个人原因在全局区中应该为“abcdefg1\0”

④ 程序执行到 p1 = getStr1();   由于个人原因在全局区中应该为“abcdefg1\0”

⑤ 程序执行到 p2 = getStr2();    由于个人原因在全局区中应该为“abcdefg1\0”

因为字符变量是一致的,所以并没有重新放在另一个内存区域,用的是同一个。

另注意:

    打印p1 p2 所指向内存空间的数据
    printf("p1:%s , p2:%s \n", p1, p2);

    打印p1 p2 的值
    printf("p1:%p , p2:%p \n", p1, p2);

1.3.2 栈区(栈区(stack) :① 由编译器自动分配释放,存放函数的参数值,局部变量的值等。② 函数运行时分配,函数结束时释放。由编译器自动分配释放 ,存放为运行函数而分配的局部变量、函数参数、返回数据、返回地址等。)

#include <stdio.h>

char *get_str(void)
{
	char str[] = "abcdedsgads"; //str在栈区,字符常量在全局区
	printf("在子函数中:str = %s\n", str);
	return str;
}

int main(vo1d)
{
	char *p = NULL;

	p = get_str();
	printf("在主函数中:p = %s\n",p);
	printf("\n");
	system("pause");
	return 0;
}

请问打印出来的内容一致吗?为什么不一致?

① 程序执行到 get_str();  包括第一次打印,到此都是正常的。

② 程序执行到 p = get_str();  讲数组的首地址赋值给p,到此也是正常的。

③ 程序运行完  p = get_str();  还未运行 printf("在主函数中:p = %s\n",p);时

     注意此时的变化:栈区(stack) :① 由编译器自动分配释放,存放函数的参数值,局部变量的值等。② 函数运行时分配,函数结束时释放。由编译器自动分配释放 ,存放为运行函数而分配的局部变量、函数参数、返回数据、返回地址等。

④ 程序运行至打印      打印出来了乱码

 

1.3.3 堆区(heap) : 一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能由操作系统回收。

接着栈所说的内容,我们该如何取出这一串字符呢?

#include <stdio.h>
#include <String.h>

char *get_str(void)
{
	char *p1 = NULL;

	p1 = (char *)malloc(100);

	if (p1 == NULL)
	{
		return NULL;
	}
	strcpy
	
	(p1,"adsagldsjg1k");
	return p1;
}


int main(vo1d)
{

	char *p = NULL;

	p = get_str();
	if (p != NULL) 
	{
		printf("在主函数中:p = %s\n",p);
		free(p);
		p = NULL;
	}
	
	printf("\n");
	system("pause");
	return 0;
}

① 程序运行至  char *p = NULL;

② 程序运行至  get_str();  中的 char *p1 = NULL;

③ 程序运行至 p1 = (char *)malloc(100);

④ 程序运行到 strcpy(p1,"adsagldsjg1k");

⑤ 程序运行到  p = get_str();

⑥ 程序运行到  printf("在主函数中:p = %s\n",p);

⑦ free(p); 只是解除了程序对于堆区地址0x20的使用权,并没有将0x20处的数据清除。

1.4 函数的调用模型

1.5 栈的生长方向和内存存放方向

#include <stdio.h>

int main(void)
{
    int a;
    int b;

    char buf[4];

    printf("&a: %p\n", &a);
    printf("&b: %p\n", &b);

    printf("buf的地址 : %p\n", &buf[0]);
    printf("buf+1地址: %p \n", &buf[1]);

    return 0;
}

 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C语言进阶 ~ 内存四区(栈、堆、全局、代码区) 的相关文章

  • Logistic回归模型(R语言)

    1 分组数据的Logistic回归模型 下面我们以一道例题来说明 xff0c R软件中实现分组数据的logistics回归模型 xff1a 代码实现如下 xff1a data10 4 lt read csv 34 C Users Admin
  • Eigen库使用指南(转载)

    Eigen库使用指南 转载 目录 Eigen库使用指南 转载 1 模块和头文件 2 Matrix类 3 矩阵与向量的运算 4 Array类 5 块操作 6 矩阵初始化 7 归约 xff0c 迭代器 xff0c 广播 8 Map类 9 混淆问
  • MAVEN SpringMVC IDEA里 修改js、html需要重启才会生效

    目的 xff1a 修改js html时可以立即生效 1 将tomcat服务配置中的before launch中的数据清空 2 修改如图 3 勾选File Settings Compiler 4 快捷键 Ctrl 43 Alt 43 Shif
  • linux命令之数据库操作

    五 数据库操作 1 mysql 重新启动 sudo etc init d mysql restart ps 看 etc mysql my conf 下的log的配置信息 一般在 var log mysql error log 2 mysql
  • Word删除单独页页眉与分节处理

    通常在word中处理页眉页脚时 xff0c 我们会发现一删全删 xff0c 一改全变的情况 xff0c 而网上方案往往难以解决且晦涩难懂 xff0c 因此 xff0c 将方法分享一下 xff0c 希望大家一看便懂 xff08 本文以遇到的三
  • Threat of Adversarial Attacks on Deep Learning in Computer Vision: A Survey 论文阅读笔记

    本文是论文的阅读笔记 Paper A Threat of Adversarial Attacks on Deep Learning in Computer Vision A Survey Author Naveed Akhtar cor n
  • ubuntu20.04安装skopeo

    文章目录 1 先安装好go环境2 配置环境3 安装skopeo3 1 安装依赖3 2 获取项目 xff0c 并进行编译3 3 构建文档3 4 安装3 5 验证 4 注意 1 先安装好go环境 如何安装go环境 xff0c 最好选择系统范围内
  • go get得到的东西不在$GOPATH中

    最近在ubuntu20 04上安装oci image tool xff0c 按照官方文档执行 xff0c 结果第一步就发现go get得到的东西找不到 xff0c 后来查阅了这篇博客 xff0c 发现了原因 xff0c 是因为开了gopro
  • 容器镜像加密-containerd imgcrypt实践

    最近在研究容器镜像加密 xff0c 发现国内对容器镜像这部分的博客好像不太多 xff0c 在看了一些人的博客后 xff0c 跟着他们的步骤进行了containerd imgcrypt的实践 xff0c 期间出现了一些错误 xff0c 决定记
  • ubuntu20.04安装howdy

    howdy是一款用于Linux系统的人脸识别身份验证系统 xff0c 可以看作是Windows中的Windows Hello的替代品 xff0c howdy在安装的过程中会去外网下载一些安装包 xff0c 所以前提条件是有访问外网的手段 h
  • zipimport.ZipImportError: can‘t decompress data; zlib not available

    在Ubuntu16 04上通过pyenv安装python3 6 8时出现以下错误 xff0c 记录一下 ubuntu 64 ubuntu pyenv span class token function install span 3 6 8
  • Ubuntu16.04安装zabbix4.0

    以下教程为Ubuntu16 04安装zabbix4 0 xff0c 需提前自己安装好Mysql数据库 其中第1 7点为主节点安装配置Zabbix server Zabbix agent Web前端 xff0c 第8点不用进行 xff1b 第
  • Ubuntu16.04搭建gitea1.14.1

    以下教程为在Ubuntu16 04上搭建gitea1 14 1 xff0c 同时附上官方教程链接 文章目录 1 数据库准备1 1 登录数据库1 2 创建gitea用户1 3 创建gitea数据库1 4 给gitea用户赋予数据库的权限1 5
  • ubuntu16.04搭建spice-html5用于配合KVM

    文章目录 0 配置需求1 启动spice客户端2 配置websockify3 配置spice html54 浏览器访问 0 配置需求 Firefox或Chrome浏览器 xff0c IE浏览器也可以 xff0c 但是效果不是太好 WebSo
  • python 数据挖掘中的数值计算

    一 环境安装 环境配置 xff1a OS xff1a Red Hat 4 4 7 11 查看命令 xff1a uname a xff1a 电脑以及操作系统的相关信息 cat proc version xff1a 正在运行的内核版本 cat
  • 银河麒麟下libguestfs-tools中virt工具无法使用情况

    银河麒麟下libguestfs tools中virt工具无法使用情况 问题描述 xff1a 在银河麒麟高级服务器版本V10下安装libguestfs tools工具包后 xff0c 使用virt xx等一系列命令都会出现以下错误 xff1a
  • ubuntu16.04搭建containerd

    本博客具体介绍在ubuntu16 04下安装containerd的过程 xff0c 有关ctr的命令都要有root权限才能运行 搭建过程 下载containerd安装包解压containerd安装包到根目录下启动containerd并设置开
  • KVM安装Windows11系列(一)

    本教程系列为KVM安装Windows11 xff0c 会分成两部分 xff0c 第一部分会跳过Windows11的硬件要求TPM和安全启动 xff0c 第二部分会安装TPM模拟器进行模拟 文章目录 软件环境下载Windows11镜像和驱动创
  • 容器技术对比(Docker/LXC/LXD/Multipass)

    DockerLXCLXDMultipass基本介绍为了打破 程序即应用 的观念 xff0c 通过镜像imges将作业系统核心除外 xff0c 运作应用程序所需的系统环境 xff0c 由下而上打包 xff0c 达到应用程序跨平台间的无缝接轨运
  • KVM虚拟机配置静态IP(一):Ubuntu16.04

    该系列文章为制作KVM虚拟机镜像的同时配置静态IP xff0c 为用户直接提供好固定IP xff0c 无需手动配置 系列文章目录 Ubuntu16 04Ubuntu18 04Centos6Centos7 文章目录 系列文章目录软件环境一 安

随机推荐

  • KVM虚拟机配置静态IP(二):Ubuntu18.04

    该系列文章为制作KVM虚拟机镜像的同时配置静态IP xff0c 为用户直接提供好固定IP xff0c 无需手动配置 系列文章目录 Ubuntu16 04Ubuntu18 04Centos6Centos7 文章目录 系列文章目录软件环境一 安
  • KVM虚拟机配置静态IP(三):Centos6

    该系列文章为制作KVM虚拟机镜像的同时配置静态IP xff0c 为用户直接提供好固定IP xff0c 无需手动配置 系列文章目录 Ubuntu16 04Ubuntu18 04Centos6Centos7 文章目录 系列文章目录软件环境一 安
  • KVM虚拟机配置静态IP(四):Centos7

    该系列文章为制作KVM虚拟机镜像的同时配置静态IP xff0c 为用户直接提供好固定IP xff0c 无需手动配置 系列文章目录 Ubuntu16 04Ubuntu18 04Centos6Centos7 文章目录 系列文章目录软件环境一 安
  • KVM下Ubuntu18.04打开设置注销问题

    问题描述 xff1a 在KVM中创建Ubuntu18 04 xff0c 打开系统设置 xff0c 发现直接注销 xff08 不是锁屏 xff0c 因为所有程序都退出 xff09 解决思路 在google上搜索发现大多数都是指向显卡问题 xf
  • cloud-init中NoCloud配置

    本文章主要记录cloud init工具中NoCloud数据源的使用方法 xff0c 可以搭配KVM镜像制作系列文章 xff0c 为用户定制操作系统 文章目录 NoCloud使用方法1 安装并初始化文件2 修改cloud init配置文件3
  • winform界面设计

    来自于以下两个地址 xff0c 为便于查阅 xff0c 所以全复制到了自己的BLOG xff1a http dotnet chinaitlab com VCNET 436373 html http www aspxboy com priva
  • cloud-init离线安装编程环境

    本博客主要介绍通过cloud init工具实现在Ubuntu16 04操作系统和KVM虚拟化技术下实现创建虚拟机同时离线安装编程环境 文章目录 1 准备离线安装包1 xff09 下载软件包2 xff09 创建放置软件包的磁盘3 xff09
  • openEuler22.03安装zabbix4.0

    以下教程为openEuler22 03安装zabbix4 0 xff0c 主要原因是openEuler官方和zabbix官方提供的提供的软件源中没有相关软件 xff0c 因此需要使用zabbix源码进行编译 xff0c 并且安装过程中会出现
  • 1.2 SingleThreadExecutor

    线程池工具类给我们提供了一些常见的线程池 xff0c 这篇来谈一谈SingleThreadExecutor线程池 使用方式 创建方式比较简单 xff0c 直接使用工具创建就ok xff0c Executors newSingleThread
  • pycharm终止代码运行时报错:进程已结束,退出代码137 (interrupted by signal 9: SIGKILL)

    在pycharm中调试代码 xff0c 终止时报错 xff1a 进程已结束 退出代码137 interrupted by signal 9 SIGKILL 网上查找时 xff0c 遇到这个问题的一般是训练网络时的内存不足 xff0c 进程被
  • 【多机多卡】mmsegmentation训练报错“RuntimeError: NCCL error in: /opt/pytorch/pytorch/torch/csrc/distributed/”

    多机多卡训练代码 xff1a 报错信息 xff1a RuntimeError NCCL error in opt pytorch pytorch torch csrc distributed c10d ProcessGroupNCCL cp
  • STM32F4工程--串口--配置一个发送函数(详细版)

    STM32F4工程 串口 配置一个发送函数 xff08 库函数 xff09 芯片 xff1a STM32F429IGT6 目录 一 初始化串口相关的参数 二 初始化串口IO口时钟等参数 三 函数声明 四 主函数 xff08 执行函数 xff
  • Ubuntu:NVIDIA-SMI has failed because it couldn‘t communicate with the NVIDIA driver. 解决方法总结

    在Ubuntu上运行Cuda并行计算的渲染项目 xff08 Massively Parallel Rendering of Complex Closed Form Implicit Surfaces 论文代码GUI部分源码 xff09 xf
  • Ubuntu16.04下基于BUCK安装onos

    踩了很多坑 xff0c 其中很多错误也没整明白怎么回事 xff0c 实在搞不了就重装系统 xff0c 经过多次测试 xff0c 找到了一个合适的安装步骤安装ONNO 1 13 2 1 安装mininet 需要可以安装 xff09 此步骤安装
  • Ubuntu16.04安装中文输入法

    转载自 xff1a https jingyan baidu com article 86f4a73e8f534637d752695e html 这是基于Fcitx框架的 可以安装Google pinyin xff0c Sougou piny
  • 基于IDEA分析ONOS源码

    1 安装Java依赖 sudo apt get install software properties common y amp amp sudo add apt repository ppa webupd8team java y amp
  • ESP8266- 使用AT指令获取网络时间

    前言 xff1a 很早就考虑过用 ESP8266 获取网络时间 xff0c 以前都是用 ESP8266 刷机智云的 Gagent 固件 xff0c 但无奈现在手头的 ESP 01 的 Flash 只有 1M xff0c 实在无法胜任 经过在
  • 使用git在项目中的一些经验

    近期在使用gitLab管理项目 xff0c 在使用过程中遇到一些问题 xff0c 在此整理一下 git的基本配置在这里就不累赘了 xff0c 主要讲一下如何将项目托管到gitLab xff0c 并实现多人协作开发 尤其介绍一下如何解决冲突
  • Ubuntu安装ROS报错 sudo: rosdep:找不到命令

    安装ROS时初始化rosdep过程中 xff0c 执行到 xff1a sodu rosdep init 报错 xff1a sudo rosdep xff1a 找不到命令 原因 xff1a 没有安装python rosdep这个包 解决方法
  • C语言进阶 ~ 内存四区(栈、堆、全局、代码区)

    特别声明 xff1a 该部分是根据B站大佬 什么都想干好的视频学习而来 目录 1 1 数据类型本质分析 1 1 1 数据类型概念 1 1 2 数据类型的本质 1 1 3 数据类型的别名 1 1 4 数据类型之 void 1 2 变量的本质分