04.函数与结构体
函数与结构体
本章讲解C语言中函数与结构体的核心概念。函数通过定义、调用与参数传递实现代码复用与逻辑分离。递归用于解决分治问题,需注意栈溢出风险。结构体封装多种数据类型,定义自定义对象,支持成员函数与运算符重载,提升代码可读性与维护性。实例演示函数、递归与结构体的应用,帮助掌握模块化编程与复杂数据管理。
函数基本概念
什么是函数
- 一段具有特定功能的代码块
- 可以执行某项特定任务并返回结果
例如输入a
与b
求a+b
这件事,不使用函数:
1 |
|
使用函数:
1 |
|
函数的定义
函数的定义:
- 返回值类型:定义返回类型,
void
则表示不返回内容; - 函数名:用该名字调用函数;
- 参数类型: \(0\)个或多个参数,参数之间用“
,
”隔开; - 参数名:每个参数类型之后跟参数名,在函数里可以使用这些参数;
- 函数内逻辑:执行特定任务;
- 返回值:如果返回值类型不是
void
,则必须用return
返回一个变量的值。
1 | <返回值> <函数名>(<参数类型1> <参数名1>, <参数类型2> <参数名2>, ...) { |
如果<返回值>
不是void
,则函数结尾必须return <返回值>
无返回值的函数示例:
1 | void Hello() { |
注意事项:
- 推荐将自定义函数写在
int main
的上面,否则编译器在执行main()
的时候不知道这个函数; - 函数的名称和变量名都是大小写敏感的,定义
Add()
就不能用add()
去调用;
函数的调用与参数传递
函数通过函数名调用,格式是
<函数名> (<参数1>, <参数2>, ...);
- 没有返回值的函数,直接调用
1
2
3
4int main() {
Hello();
return 0;
} - 有返回值的函数,将返回值赋值给变量,或直接使用返回值
1
2
3
4
5
6int main() {
// ...
int c = Add(a, b);
printf("%d\n", Add(a, b));
return 0;
} - 参数要与变量类型对应,定义
int
则对应传入int
,定义double
则对应传入double
1
2
3
4
5
6
7
8
9int Add(double a, double b) {
return a + b;
}
int main() {
double x, y;****
// scanf ...
printf("%f\n", Add(x, y));
// 错误示例:char x, y; Add(x, y); Add的参数是double,不能传char参数
}
函数内对参数的修改,不会影响到调用函数所传的变量
尝试
1 | void Change(int a) { |
输出为
1 | Test1: 10 |
可以这样认为,将main
中的x
传入Change
函数时,函数得到的是x
的一个副本a
,对a
的任何修改都是对副本的修改,不会影响x
本来的值。
这里传给函数的x
称为“实际参数”,函数得到的a
称为“形式参数”。
在C++
中,如果想让函数得到变量“本体”,可使用叫做“引用”的语法,即在参数前加“&
”符号:
1 | void Change(int &a) { |
输出为
1 | Test1: 10 |
两份代码唯一区别是
Change
函数定义时,其参数定义由int a
改为了int &a
,表示对传入参数的引用,这样对a
的修改会对本体x
生效。
函数的返回可以“截断”后续的逻辑
1 | int f(int a, int b) { |
当满足
a<b
时,return a
的逻辑会被执行,函数中任何return
被执行时,之后的代码都不会再被执行。
当不满足a<b
时,代码会继续向下执行到return b
。
函数的意义
- 复用同一段代码处理相同的任务
- 让程序更加模块化,更易于理解和维护
例:分别求 \(1\sim 10\)、 \(200\sim 300\)的和并输出
不使用函数:
1 | int main() { |
使用函数精简了代码,且逻辑清晰。
如果这时任务改成,“求一段数从第一个数开始,奇数位置上的和”,如果没有使用函数,则每个执行此任务的地方都要修改代码。当使用函数时,只需要修改CalSum
这个函数内的i ++
改成i += 2
就可以,代码维护变得简单。
函数的递归
递归的定义
1 | 名词定义 递归() { |
递归即函数自己调用自己,或若干个函数在互相调用。当然上面的示例看起来永远不会结束,它是一个无限递归。
通常用递归来解决可以被分解的问题,在问题足够小的时候有一个终点。比如给出 \(n\),求 \(n\)的阶乘,即 \(n!=1\times 2\times, \dots, n\),递归可以这样写:
1 | int f(int x) { |
所以一个合理的递归,应该有两部分组成
- 对问题的分解:调用函数本身去解决一个“更小”的问题
- 这里
f(x - 1)
就是对问题的分解,递归地去处理规模更小的x-1
的阶乘,并乘以x
得到f(x)
的解
- 这里
- 递归的终点:当问题足够小时,以一个确定的逻辑返回,而不再继续分解
if(x == 0) return 1
就是在做这件事,确保递归不会无限进行下去,0
是一个足够小的问题
递归的意义
实际上,任何递归逻辑都可以用非递归的代码去完成,比如阶乘,用一个循环将 \(1\sim n\)累乘起来也能实现。
但在一些复杂问题中,递归能帮助我们更好地理清思路,将问题抽丝剥茧,要比从全局思考正确步骤更为容易。
以汉诺塔问题为例,有三根柱子,一根柱子上有 \(n\)个大小不同的盘子,将所有的盘子移动到另一根柱子上,在移动过程中,任何时候都不能让大盘子在小盘子上面。

如果把柱子从左到右命名为 \(A\)、
\(B\)、 \(C\),以“A to C
”这样的指令表示把
\(A\)最顶部的盘子挪到 \(C\)的最顶部,当一开始 \(A\)柱有 \(n\)个盘子,以一系列指令给出将所有盘子在约束条件下从
\(A\)移到 \(C\)的方案。
不用递归的话,面对这个问题还是很懵的,那么定义一个这样的函数:
1 | void Hanoi(int n, char x, char y, char z) { |
表示把n
个盘子从变量x
所表示的柱子,利用y
作为辅助柱子,以符合题目要求的方式全部移动到z
所表示的柱子上。
那么第一次调用这个函数应该是这样:
1 | Hanoi(n, 'A', 'B', 'C'); |
可以这样分解该问题:
- \(A\)上的 \(n-1\)个盘子通过 \(C\)先移到 \(B\)
- \(A\)上最底部的那个盘子直接移到
\(C\),即输出“
A to C
” - \(B\)上当前的 \(n-1\)个盘子通过 \(A\)移到 \(C\), \(C\)上在第2步放的那个最大的盘子完全不影响这 \(n-1\)个盘子的任何移动
这3步就完成了整个任务。而第1步和第3步是相同的问题,只是问题规模减小了1,且起点柱子和终点柱子不同了而已,他们可以递归地调用Hanoi
完整的递归代码这样实现:
1 | void Hanoi(int n, int x, int y, int z) { |
递归栈
当一个函数被调用或被递归地调用,调用结束时代码还要继续往下进行,操作系统怎么“恢复现场”呢?
比如
1 | int f(int x) { |
这就有一个“栈内存”的概念,每当调用函数,在进入函数执行前,系统会“保护现场”,把所有信息保存到一块内存中。当函数执行完毕,会从这块栈内存把所有信息读回来“恢复现场”并继续执行接下来的代码。
通常系统的栈内存有一个固定大小,当递归时,每次调用都会“保护现场”,占用一些栈内存,那么递归得太深,就可能把栈内存用尽,程序就会出错,这种情况称为“栈溢出”。
比如递归求一个很大的阶乘
1 |
|
因为现代编译器会做一些优化来尽可能避免栈溢出,这里作为栈溢出的例子增加了一些无用代码来确保溢出不会被优化掉,这方面的知识大家以后会了解。
栈溢出的知识点告诉我们,在思考一个问题解法的时候,如果需要的递归深度离谱的高,就要慎重考虑一下是不是最好的方案了。
当然还有一个更常见的情况,也是建议,较大的数组放在main
函数外面
1 | // 推荐做法: |
函数内的一切变量都开在“栈内存”中,过大会导致栈溢出。而函数外的变量会开在“堆内存”中,这就是其他话题了,此处不再赘述。
结构体
结构体的概念
将多种变量封装到一起从而自定义一种类型。
一个“学生”有学号、年龄、身高、体重等信息,如果一个程序想以“学生”作为对象来做数据处理,或者以某种条件筛选学生并返回学生信息:
不使用结构体
1 | // 输出学生信息 |
这还只是简单的需求。如果一个学生要记录更多维度的信息如姓名、住址、手机号等等等等,代码的编写将十分繁琐且不便维护。
将一个学生作为一个统一的对象,把信息打包封装:
1 | struct Student{ |
这样Student
就可以作为一个自定义的数据类型,去定义变量或数组,每个该类型的变量都会包含“学生”这一概念的各种信息。
用结构体定义的变量称为该结构体的“对象”,用“.
”来调用对象内的变量。
成员函数
如果一些功能只与结构体自身有关,可以理解为这个结构体自身的某种功能,那么可以将函数定义为结构体的“成员函数”,同样用“.
”来调用对象内的成员函数,成员函数可以直接使用和修改对象内的变量。
1 | struct Student{ |
运算符的重载
int
、double
这些类型天然具备的
+,-,*,/,>,<
等操作,作为自定义变量类型的结构体并没有良好的定义。
我们当然可以用函数实现结构体之间的各种运算,但如果能用这些运算符,会让代码逻辑更为清晰。
比如两个学生的大小用年龄的大小关系来定义,函数可以这样实现:
1 | struct Student{ |
而重载运算符,就可以在结构体的对象之间执行相应的运算,比如
1 | struct Student{ |