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 语言中初始化指针数组?

指针变量可以在声明时初始化,方法是将其赋给现有变量的地址。以下代码片段展示了如何初始化指针 -

int x = 10;
int *y = &x;

默认情况下,所有变量(包括指针变量)都属于"自动存储类"。这意味着指针变量将存储不可预测的、垃圾的、随机的内存地址,这可能导致未定义的行为和程序的潜在风险,例如段错误。因此,如果在声明时没有指定要存储的特定值,则应将其初始化为 NULL。

int *ptr = NULL;

"指针数组"将地址存储在每个元素中。数组的类型必须与目标变量的类型匹配。

使用 static 关键字初始化指针数组

您还可以使用 static 关键字 初始化 指针数组,在每个下标中存储"0"。

示例

#include <stdio.h>

int main(){

   static int *ptr[5];

   for (int i = 0; i < 5; i++){
      printf("ptr[%d] = %d
", i, ptr[i]); 
   }
   
   return 0;
}

输出

运行代码并检查其输出 −

ptr[0]= 0
ptr[1]= 0
ptr[2]= 0
ptr[3]= 0
ptr[4]= 0

初始化整数指针数组

这里,我们声明一个整数指针数组,并存储三个整数变量的地址。

示例

#include <stdio.h>

int main(){

   int a = 10, b = 20, c = 30;
   int *ptr[3] = {&a, &b, &c};

   for (int i = 0; i < 3; i++){
      printf("ptr[%d]: address: %d value: %d
", i, ptr[i], *ptr[i]);
   }
   
   return 0;
} 

输出

运行代码并检查其输出 −

ptr[0]: address: 6422040 value: 10
ptr[1]: address: 6422036 value: 20
ptr[2]: address: 6422032 value: 30

通过直接地址初始化指针数组

我们可以将普通数组中每个元素的地址存储在指针数组的相应元素中。

示例

#include <stdio.h>

int main(){

   int arr[] = {10, 20, 30};
   int *ptr[3] = {&arr[0], &arr[1], &arr[2]};

   for (int i = 0; i < 3; i++){
      printf("ptr[%d]: address: %d value: %d
", i, ptr[i], *ptr[i]);
   }
   
   return 0;
}

输出

运行代码并检查其输出 −

ptr[0]: address: 6422032 value: 10
ptr[1]: address: 6422036 value: 20
ptr[2]: address: 6422040 value: 30

通过基址遍历数组

当我们获取数组的基址(在本例中为" &arr[0]" )时,我们可以获取其后续元素的地址,因为指针会根据数据类型的大小递增。

因此,仅凭基址(数组名称与第 0 个元素的地址相同),我们就可以遍历数组。

示例 1

请看以下示例 -

#include <stdio.h>

int main(){

   int arr[] = {10, 20, 30};
   int *ptr=arr;

   for (int i = 0; i < 3; i++){
      printf("ptr[%d]: address: %d value: %d
", i,ptr+i, *(ptr+i));
   }
   
   return 0;
}

输出

运行代码并检查其输出 −

ptr[0]: address: 6422020 value: 10
ptr[1]: address: 6422024 value: 20
ptr[2]: address: 6422028 value: 30

示例 2:使用指针数组遍历二维数组

本例中,我们有一个二维数组。每行第 0 个元素的地址存储在一个指针数组中。遍历时,指针数组中每个元素的地址(指向相应行的第 0 个元素)都会递增,以提取每行的值。

#include <stdio.h>

int main(){

    // 二维数组
    int arr[3][4] = {
        {1, 2, 3, 4},
        {5, 6, 7, 8},
    };
    
    int ROWS = 2, COLS = 4;
    int i, j;
    
    // 指针
    int (*ptr)[4] = arr;
    
    // 通过指针 ptr 打印数组元素
   	for (i = 0; i < ROWS; i++) {
      for (j = 0; j < COLS; j++) {
         printf("%d ", *(ptr[i]+j));
      }
      printf("
");
   }
   
   return 0;
}

输出

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

1 2 3 4 
5 6 7 8

示例 3

这里我们实际上不需要指针数组,因为我们可以使用这个二维数组的名称作为其基指针,并按行和列递增它以获取给定二维数组中的元素 -

#include <stdio.h>

int main(){

    // 二维数组
    int arr[3][4] = {
        {1, 2, 3, 4},
        {5, 6, 7, 8},
    };
    
    int ROWS = 2, COLS = 4;
    int i, j;
    
    // 指针
    int *ptr = arr;
    
    // 通过指针 ptr 打印数组元素
   for (i = 0; i < ROWS; i++){
      for (j = 0; j < COLS; j++){
         printf("%d  ", *(ptr + i * COLS + j));
      }
      printf("
");
   }
   
   return 0;
}

输出

输出类似于前面的代码 -

1 2 3 4
5 6 7 8

初始化字符指针数组(字符串)

在 C 语言编程中,字符串是 char 数据类型的数组。由于数组名称也代表其第 0 个元素的地址,因此字符串可以声明为 -

char arr[] = "Hello";

使用指针表示法,将字符串赋值给 char 指针,如下所示 -

char *arr = "Hello";

然后,我们可以声明一个字符指针数组来存储多个字符串,如下所示 -

char *arr[3] = {"string1", "string2", "string3", . . . };

示例

以下示例包含一个字符指针数组,用于存储计算机语言的名称 -

#include <stdio.h>

int main(){

   char *langs [10] = {
      "PYTHON", "JAVASCRIPT", "PHP",
      "NODE JS", "HTML", "KOTLIN", "C++",
      "REACT JS", "RUST", "VBSCRIPT"
   };

   for(int i = 0; i < 10; i++)
      printf("%s
", langs[i]);

   return 0;
}

输出

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

PYTHON
JAVASCRIPT
PHP
NODE JS
HTML
KOTLIN
C++
REACT JS
RUST
VBSCRIPT

在此程序中,"langs"是一个指向一个包含 10 个字符串的数组的指针。因此,如果"langs[0]"指向地址 5000,那么"langs + 1"将指向地址 5004,该地址存储了指向第二个字符串的指针。

因此,我们也可以使用以下循环变体来打印字符串数组 -

for (int i = 0; i < 10; i++){
   printf("%s
", *(langs + i));
}

动态指针数组的初始化

您可以使用 malloc() 函数 以动态方式声明和初始化指针数组。

示例

请看以下示例 -

#include <stdio.h>

int main(){

   int *arr = (int *)malloc (sizeof (int) * 5); 

   for(int i = 0; i < 5; i++){
      arr[i] = i;
   }
   for (int x = 0; x < 5; x++){
      printf("%d %d
", x, arr[x]);
   }
   
   return 0;
}

输出

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

0 0
1 1
2 2
3 3
4 4

您甚至可以请求用户输入并将值分配给数组指针中的元素 -

for(i = 0; i < 5; i++){
   scanf("%d", &x);
   arr[i] = x;
}