跳转到内容
主菜单
主菜单
移至侧栏
隐藏
导航
首页
最近更改
随机页面
MediaWiki帮助
代码酷
搜索
搜索
中文(中国大陆)
外观
创建账号
登录
个人工具
创建账号
登录
未登录编辑者的页面
了解详情
贡献
讨论
编辑“︁
C 语言函数指针
”︁
页面
讨论
大陆简体
阅读
编辑
编辑源代码
查看历史
工具
工具
移至侧栏
隐藏
操作
阅读
编辑
编辑源代码
查看历史
常规
链入页面
相关更改
特殊页面
页面信息
外观
移至侧栏
隐藏
您的更改会在有权核准的用户核准后向读者展示。
警告:
您没有登录。如果您进行任何编辑,您的IP地址会公开展示。如果您
登录
或
创建账号
,您的编辑会以您的用户名署名,此外还有其他益处。
反垃圾检查。
不要
加入这个!
{{Note|本教程假设读者已掌握基础C语言知识(变量、函数、指针基础),若需复习请先学习[[C语言指针基础]]章节}} = 函数指针(Function Pointers)概念 = '''函数指针'''是指向函数的指针变量,它存储的是函数的入口地址而非数据。在C语言中,函数名本身就是指向该函数代码的指针。 == 核心特性 == * 允许运行时动态选择调用的函数 * 可作为参数传递给其他函数(回调机制) * 能存储在数组中形成"函数表" * 类型安全性:必须匹配返回类型和参数列表 <mermaid> graph LR A[普通指针] -->|指向| B[数据地址] C[函数指针] -->|指向| D[可执行代码] </mermaid> = 基本语法 = == 声明格式 == <pre> 返回类型 (*指针变量名)(参数列表); </pre> '''示例声明对比''': {| class="wikitable" |- ! 类型 !! 示例 |- | 普通函数 | <code>int func(int x);</code> |- | 函数指针 | <code>int (*funcPtr)(int);</code> |} == 初始化与调用 == <syntaxhighlight lang="c"> #include <stdio.h> int add(int a, int b) { return a + b; } int main() { // 声明并初始化函数指针 int (*operation)(int, int) = &add; // &可选 // 通过指针调用函数 int result = operation(3, 5); printf("3 + 5 = %d\n", result); // 输出: 3 + 5 = 8 return 0; } </syntaxhighlight> = 高级应用 = == 回调函数(Callback)== 函数指针最常见的用途是实现回调机制: <syntaxhighlight lang="c"> #include <stdio.h> // 回调函数类型定义 typedef void (*Callback)(int); void processArray(int arr[], int size, Callback cb) { for (int i = 0; i < size; i++) { cb(arr[i]); // 对每个元素调用回调 } } void printElement(int x) { printf("%d ", x); } void squareElement(int x) { printf("%d ", x * x); } int main() { int nums[] = {1, 2, 3, 4, 5}; printf("Original: "); processArray(nums, 5, printElement); printf("\nSquared: "); processArray(nums, 5, squareElement); return 0; } </syntaxhighlight> '''输出''': <pre> Original: 1 2 3 4 5 Squared: 1 4 9 16 25 </pre> == 函数指针数组 == 可创建函数指针数组实现状态机或命令模式: <syntaxhighlight lang="c"> #include <stdio.h> typedef void (*Command)(); void start() { printf("Starting system...\n"); } void stop() { printf("Stopping system...\n"); } void reboot() { printf("Rebooting...\n"); } int main() { Command commands[] = {start, stop, reboot}; int choice; printf("Menu:\n1. Start\n2. Stop\n3. Reboot\n> "); scanf("%d", &choice); if (choice >= 1 && choice <= 3) { commands[choice - 1](); // 数组索引从0开始 } return 0; } </syntaxhighlight> = 复杂类型解析 = 使用<code>typedef</code>简化复杂声明: <syntaxhighlight lang="c"> // 普通声明 int (*complexPtr)(int (*)(int), int); // 使用typedef简化 typedef int (*UnaryOp)(int); typedef int (*BinaryOp)(UnaryOp, int); BinaryOp simplerPtr; // 等同于complexPtr </syntaxhighlight> = 真实案例:qsort函数 = C标准库中的<code>qsort</code>是函数指针的经典应用: <syntaxhighlight lang="c"> #include <stdio.h> #include <stdlib.h> // 比较函数 int compareInt(const void *a, const void *b) { return (*(int*)a - *(int*)b); } int main() { int arr[] = {5, 2, 8, 1, 4}; int n = sizeof(arr)/sizeof(arr[0]); qsort(arr, n, sizeof(int), compareInt); for (int i = 0; i < n; i++) printf("%d ", arr[i]); // 输出: 1 2 4 5 8 return 0; } </syntaxhighlight> = 数学表示 = 函数指针类型可以表示为: <math> F : T_1 \times T_2 \times \dots \times T_n \rightarrow R </math> 其中: * <math>T_i</math> 是第i个参数类型 * <math>R</math> 是返回类型 = 常见问题 = {{Warning|1=函数指针易错点}} * '''类型不匹配''':参数列表或返回类型不一致 * '''NULL指针调用''':未初始化就调用 * '''错误转换''':强制类型转换可能导致未定义行为 = 性能考量 = * 现代CPU对函数指针调用有良好优化 * 比switch-case更高效的条件分支实现方式 * 间接调用可能影响内联优化 = 扩展阅读 = * [[C语言多态实现]] * [[动态链接库(DLL)原理]] * [[面向对象C编程]] [[Category:编程语言]] [[Category:C]] [[Category:C 语言指针]]
摘要:
请注意,所有对代码酷的贡献均被视为依照知识共享署名-非商业性使用-相同方式共享发表(详情请见
代码酷:著作权
)。如果您不希望您的文字作品被随意编辑和分发传播,请不要在此提交。
您同时也向我们承诺,您提交的内容为您自己所创作,或是复制自公共领域或类似自由来源。
未经许可,请勿提交受著作权保护的作品!
取消
编辑帮助
(在新窗口中打开)
该页面使用的模板:
模板:Mbox
(
编辑
)
模板:Note
(
编辑
)
模板:Warning
(
编辑
)
模块:Arguments
(
编辑
)
模块:Message box
(
编辑
)
模块:Message box/ambox.css
(
编辑
)
模块:Message box/configuration
(
编辑
)
模块:Yesno
(
编辑
)