【C 语言系统入门教程】第 14 讲深入理解指针 (4) | 零基础学习笔记前言本讲是指针进阶收官篇聚焦字符指针、数组指针、二维数组传参、函数指针、函数指针数组、转移表六大高阶指针知识点彻底打通 C 语言指针的最后壁垒是理解复杂指针、底层编程、函数回调的核心一课。系列往期笔记第 1 讲C 语言常见概念第 2 讲C 语言数据类型和变量第 3 讲分支和循环上第 4 讲分支和循环下第 5 讲数组第 6 讲函数第 7 讲数组和函数实践扫雷游戏第 8 讲VS 实用调试技巧第 10 讲操作符详解第 11 讲深入理解指针 (1)第 12 讲深入理解指针 (2)第 13 讲深入理解指针 (3) 本讲学习目标掌握字符指针的两种用法理解常量字符串存储特性。区分指针数组与数组指针掌握数组指针的定义与使用。吃透二维数组传参的本质掌握形参的两种写法。掌握函数指针的定义、赋值与调用。学会用typedef重命名复杂指针类型。掌握函数指针数组的定义与使用场景。理解转移表思想用函数指针数组简化多分支代码。 核心学习内容1. 字符指针变量字符指针char*除指向单个字符外还可指向常量字符串首字符地址。1.1 两种使用方式// 1. 指向单个字符 char ch w; char* pc ch; // 2. 指向常量字符串存首字符地址 const char* pstr hello bit.;核心const char* pstr hello bit.不是把字符串存入指针而是把字符串首字符地址存入pstr。1.2 常量字符串笔试题#include stdio.h int main() { char str1[] hello bit.; char str2[] hello bit.; const char *str3 hello bit.; const char *str4 hello bit.; ​ if(str1 str2) printf(str1 and str2 are same\n); else printf(str1 and str2 are not same\n); ​ if(str3 str4) printf(str3 and str4 are same\n); else printf(str3 and str4 are not same\n); return 0; }结果str1 and str2 are not same str3 and str4 are same原因数组初始化会开辟独立内存str1、str2 地址不同。常量字符串存放在只读内存区str3、str4 指向同一块地址。字符数组与字符指针存储对比图​2. 数组指针变量2.1 数组指针 vs 指针数组写法类型含义int* p[10]指针数组数组存放 10 个int*指针int (*p)[10]数组指针指针指向含 10 个 int 的数组核心[]优先级高于*加()保证先与*结合。2.2 数组指针定义与初始化数组指针用于存放整个数组的地址数组名。int arr[10] {0}; // 数组指针指向int[10]类型的数组 int (*p)[10] arr;类型解析int (*p)[10]p数组指针变量名*说明是指针[10]指向的数组包含 10 个元素int数组元素类型数组指针指向数组​3. 二维数组传参的本质核心结论二维数组名是第一行一维数组的地址类型为int (*)[5]数组指针。3.1 形参两种写法// 写法1二维数组形式 void test(int a[3][5], int r, int c); ​ // 写法2数组指针形式本质 void test(int (*p)[5], int r, int c);3.2 数组指针访问二维数组#include stdio.h void test(int (*p)[5], int r, int c) { int i 0, j 0; for(i0; ir; i) { for(j0; jc; j) { // *(*(pi)j) 等价于 p[i][j] printf(%d , *(*(pi)j)); } printf(\n); } } int main() { int arr[3][5] {{1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7}}; test(arr, 3, 5); return 0; }4. 函数指针变量4.1 函数地址函数名 函数名都是函数的地址。#include stdio.h void test() { printf(hehe\n); } int main() { printf(%p\n, test); // 函数地址 printf(%p\n, test); // 函数地址 return 0; }4.2 函数指针定义// 函数 int Add(int x, int y) { return xy; } // 函数指针指向返回int、参数(int,int)的函数 int (*pf)(int, int) Add;类型解析int (*pf)(int, int)pf函数指针变量名*说明是指针(int,int)函数参数类型int函数返回值类型4.3 函数指针调用// 两种调用方式等价 printf(%d\n, (*pf)(2,3)); printf(%d\n, pf(2,3));4.4 typedef 重命名函数指针简化复杂函数指针类型// 重命名 void(*)(int) 为 pfun_t typedef void(*pfun_t)(int); pfun_t signal(int, pfun_t);5. 函数指针数组定义存放函数指针的数组要求数组中函数的返回值、参数完全一致。// 函数指针数组存放4个运算函数 int (*p[5])(int, int) {0, add, sub, mul, div};解析p先与[5]结合是数组数组元素类型int (*)(int,int)函数指针6. 转移表转移表用函数指针数组替代switch-case简化多分支函数调用。6.1 计算器转移表实现#include stdio.h // 1. 将 div 改名为 my_div 避免标准库冲突 int add(int a, int b) { return a b; } int sub(int a, int b) { return a - b; } int mul(int a, int b) { return a * b; } int my_div(int a, int b) { return a / b; } int main() { int x, y, input 1, ret 0; // 函数指针数组转移表 (对应 0:占位, 1:add, 2:sub, 3:mul, 4:div) int (*p[5])(int, int) {0, add, sub, mul, my_div}; while (input) { printf(1:add 2:sub\n3:mul 4:div\n0:exit\n请选择:); // 4. 增加 scanf 返回值检查防止非数字输入导致死循环 if (scanf(%d, input) ! 1) { // 清空输入缓冲区 while(getchar() ! \n); printf(输入无效请输入数字\n\n); continue; } if (input 1 input 4) { printf(输入操作数:); if (scanf(%d%d, x, y) ! 2) { while(getchar() ! \n); printf(输入无效请输入两个整数\n\n); continue; } // 3. 除法前增加除零检查 if (input 4 y 0) { printf(错误除数不能为零\n\n); continue; } // 2. 修正函数调用参数 ret p[input](sslocal://flow/file_open?urlx%2Cyflow_extraeyJsaW5rX3R5cGUiOiJjb2RlX2ludGVycHJldGVyIn0); printf(ret%d\n\n, ret); } else if (input 0) { printf(退出计算器\n); } else { printf(选择错误\n\n); } } return 0; }转移表函数调用示意图​ 课后习题一、选择题const char* p abcp 存储的是A. 字符串全部内容B. 首字符地址C. 字符串长度D. 数组地址int (*p)[10]是A. 指针数组B. 数组指针C. 函数指针D. 整型数组二维数组传参形参本质是A. 二维数组B. 一级指针C. 数组指针D. 函数指针函数指针int (*pf)(int,int)指向的函数是A. 返回 int参数两个 intB. 返回指针参数 intC. 返回数组参数 intD. 无返回值参数 int函数指针数组的核心用途是A. 存储整型数据B. 转移表简化分支C. 指向字符D. 模拟二维数组二、判断题两个字符指针指向同一个常量字符串地址相同。int* p[5]是数组指针。二维数组名是第一行一维数组的地址。函数名和 函数名都是函数地址。函数指针数组可以存放不同参数的函数指针。三、编程题用字符指针打印字符串hello C。定义数组指针指向一维数组并访问元素。用函数指针数组实现简易计算器加减乘除。 参考答案一、选择题BBCAB二、判断题√×是指针数组√√×必须返回值、参数一致三、编程题1. 字符指针打印字符串#include stdio.h int main() { const char* p hello C; printf(%s\n,p); return 0; }2. 数组指针访问数组#include stdio.h int main() { int arr[5] {1,2,3,4,5}; int (*p)[5] arr; int i0; for(i0;i5;i) { printf(%d ,(*p)[i]); } return 0; }3. 函数指针数组计算器见正文转移表代码 本讲总结字符指针指向常量字符串时存首字符地址只读字符串共享内存。数组指针int (*p)[10]指向数组存放数组名。二维数组传参本质传数组指针形参可写二维数组 / 数组指针。函数指针存放函数地址int (*pf)(int,int)可直接调用函数。typedef简化复杂指针类型命名。函数指针数组存放同类型函数指针用于转移表。转移表替代switch-case让多分支函数调用更简洁、易扩展。 版权说明本文为 C 语言系统学习原创笔记基于标准课件体系整理纯干货零基础友好未经允许禁止转载如有错误欢迎评论区指正