C 语言编程教程

C 语言 - 首页

C 语言基础

C 语言 - 概述 C 语言 - 特性 C 语言 - 发展历史 C 语言 - 环境设置 C 语言 - 程序结构 C 语言 - Hello World C - 编译过程 C - 注释 C - 标记 C - 关键字 C - 标识符 C - 用户输入 C - 基本语法 C - 数据类型 C - 变量 C - 整数提升 C - 类型转换 C - 类型转换 C - 布尔值

C 语言中的常量和文字

C - 常量 C - 字面量 C - 转义序列 C - 格式说明符

C 语言中的运算符

C - 运算符 C - 算术运算符 C - 关系运算符 C - 逻辑运算符 C - 位运算符 C - 赋值运算符 C - 一元运算符 C - 递增和递减运算符 C - 三元运算符 C - sizeof 运算符 C - 运算符优先级 C - 其他运算符

C 语言中的决策

C - 决策 C - if 语句 C - if...else 语句 C - 嵌套 if 语句 C - switch 语句 C - 嵌套 switch 语句

C 语言中的循环

C - 循环 C - While 循环 C - For 循环 C - Do...while 循环 C - 嵌套循环 C - 无限循环 C - Break 语句 C - Continue 语句 C - goto 语句

C 语言中的函数

C - 函数 C - Main 函数 C - 按值调用函数 C - 按引用调用函数 C - 嵌套函数 C - 可变参数函数 C - 用户定义函数 C - 回调函数 C - return 语句 C - 递归

C 语言中的作用域规则

C - 作用域规则 C - 静态变量 C - 全局变量

C 语言中的数组

C - 数组 C - 数组的属性 C - 多维数组 C - 将数组传递给函数 C - 从函数返回数组 C - 可变长度数组

C 语言中的指针

C - 指针 C - 指针和数组 C - 指针的应用 C - 指针运算 C - 指针数组 C - 指向指针的指针 C - 将指针传递给函数 C - 从函数返回指针 C - 函数指针 C - 指向数组的指针 C - 指向结构体的指针 C - 指针链 C - 指针 vs 数组 C - 字符指针和函数 C - NULL 指针 C - void 指针 C - 悬垂指针 C - 解引用指针 C - Near、Far 和 Huge 指针 C - 指针数组的初始化 C - 指针与多维数组

C 语言中的字符串

C - 字符串 C - 字符串数组 C - 特殊字符

C 语言的结构体和联合

C - 结构体 C - 结构体和函数 C - 结构体数组 C - 自引用结构 C - 查找表 C - 点 (.) 运算符 C - 枚举(或 enum) C - 结构填充和打包 C - 嵌套结构 C - 匿名结构和联合 C - 联合 C - Bit 位字段 C - Typedef

C 语言中的文件处理

C - 输入和输出 C - 文件 I/O(文件处理)

C 语言中的预处理器

C - 预处理器 C - #pragma 编译指示 C - 预处理器操作符 C - 宏 C - 头文件

C 语言中的内存管理

C - 内存管理 C - 内存地址 C - 存储类

C 其他主题

C - 错误处理 C - 可变参数 C - 命令执行 C - 数学函数 C - static 静态关键字 C - 随机数生成 C - 命令行参数

C 语言编程资源

C语言问题与解答答案 C语言快速指南 C语言速查表 C语言实用资源 C语言讨论


C 语言中的内存地址

C 语言中的内存地址

在 C 语言中,内存地址会在变量声明时赋值给该变量。 C 编译器将变量的值存储在内存的不同段中。

内存段

C 程序的不同元素存储在计算机内存的不同段中,内存包含以下段:

  • 文本段 − 文本段,也称为代码段或简称为文本,是程序中用于存储 C 程序对象版本的部分之一。
  • 初始化数据段 − 由以下程序段初始化的全局变量静态变量程序员在初始化数据段中分配内存。
  • 未初始化数据段 − 未初始化数据段也称为bss(代表block started by symbol)。程序加载时会为此段分配内存。在 C 程序执行之前,内核会将bss中的所有数据初始化为算术"0",并将指针初始化为空指针。
  • 堆栈 − 堆栈是一种 LIFO(后进先出)数据结构。堆栈段存储局部变量的值以及传递给函数的参数的值。它还维护函数调用返回的指针。
  • − 堆用于在运行时分配内存。所有执行动态内存分配的函数都与堆有关。

访问内存地址

C 语言中的内存地址可以通过地址 (&) 运算符访问或指定。要使用 printf() 函数 打印内存地址,需要使用 %p 格式说明符。

语法

以下是访问内存地址的语法 -

&variable_name

示例

在下面的示例中,我们声明两个变量并打印它们的内存地址 -

#include <stdio.h>

int main() {
    // 声明两个变量
    int a;
    int b;
    
    // 访问它们的内存
    // 地址并打印它们
   printf("Memory address of a is %p
", &a);
   printf("Memory address of b is %p
", &b);

   return 0;
}

C 编译器如何分配内存?

内存可以看作是一个字节数组,每个地址位于数组的索引上,并占用 1 个字节。

在 C 程序中声明变量时,C 编译器会根据变量的大小需求(取决于变量的类型)为其分配一个随机的内存位置。

声明 int 变量时 -

int x = 10;

编译器会将值赋值到一个随机的字节地址中。由于 int 类型需要 4 个字节,因此接下来的 4 个地址会被分配给它。

C 语言允许你查找已分配给变量的地址。您可以使用 %p 格式说明符打印内存位置的十六进制地址。

char x = 'A';
printf ("x 的地址: %p
", &x);

这将以十六进制格式打印"x"的地址 -

x 的地址:000000000061FE1F

示例

C 语言中的数组是连续的内存区域,用于保存多个相同数据类型(int、long、*char 等)的值。

#include <stdio.h>

int main() {

    // 初始化一个 int 数组
    int numbers[5] = {1,2,3,4,5};
    int i = 0;
    
    // 打印数组变量的地址
   printf("numbers = %p
", numbers);

   // 打印每个数组索引的地址
   do {
      printf("numbers[%u] = %p
", i, (void *)(&numbers[i]));
      i++;
   } while(i < 5);

   // 打印数组的大小
   printf("sizeof(numbers) = %lu
", sizeof(numbers));
}

输出

运行此代码时,将产生以下输出 -

numbers = 0x7fff0815c0e0
numbers[0] = 0x7fff0815c0e0
numbers[1] = 0x7fff0815c0e4
numbers[2] = 0x7fff0815c0e8
numbers[3] = 0x7fff0815c0ec
numbers[4] = 0x7fff0815c0f0
sizeof(numbers) = 20