函数
C语言中的函数分为两类:
- 库函数
- 自定义函数
库函数
为什么会有库函数?
我们在C语言编程的时候,很多基础功能并不是业务性的代码,在开发的过程中每个程序员都可能会用到,为了支持可移植性和提高程序的效率,库函数应运而生,方便程序员进行软件开发
简单总结一下,C语言常用的库函数都有:
- IO函数
- 字符串操作函数
- 字符操作函数
- 内存操作函数
- 时间/日期函数
- 数学函数
- 其他
库函数并不需要全部记住,对于库函数来说最重要的的事对照文档来掌握使用方法
常用的查询工具:MSDN
www.cplusplus.com
http:\en.cppreference.com
使用库函数,必须包含#include对应的头文件
自定义函数
自定义函数和库函数一样,有函数名、返回值类型和函数参数,不同的是这些都需要我们自己来设计
函数的组成:
ret_type fun_name(para1, *)
{
statement; //语句项
}
//ret_type:返回类型
//fun_name:函数名
//para1:函数参数
函数的参数
实际参数(实参):
真实传递给函数的参数。实参可以是:常量、变量、表达式、函数等。无论是哪种类型的量,在进行调用时,都必须有明确的值,以便把这些值传递给形参
形式参数(形参):
指函数名后括号中的变量。因为形式参数只有在函数被调用的过程中才实例化(分配内存单元),所以叫形式参数。形式参数实例化后相当于是实参的一份临时拷贝。形式参数当函数调用完成后就自动销毁了,因此形式参数只在函数中有效。
例:
void Swap1(int x, int y) //x,y为形参
{
int tmp = 0;
tmp = x;
x = y;
y = tmp;
}
void Swap2(int* px, int* py) //*px,*py为形参
{
int tmp = 0;
tmp = *px;
*px = *py;
*py = tmp;
}
int main()
{
int num1 = 10;
int num2 = 20;
Swap1(num1, num2); //num1,num2为实参
printf("Swap1:num1 = %d, num2 = %d\n", num1, num2);
Swap2(&num1, &num2); //num1,num2为实参
printf("Swap2:num1 = %d, num2 = %d\n", num1, num2);
return 0;
}
通过运行结构图我们发现,调用函数Swap1并没有产生我们想要的效果,这就涉及到函数的调用,也就是我们接下来要讲的
函数的调用
传值调用:
函数的形参和实参分别占有不同的内存块,对形参的改变并不会影响实参. 例如上例中的Swap1函数
传址调用:
- 传址调用就是把函数外部创建变量的 内存地址 传递给函数参数的一种调用函数的方式
- 这种传参方式让函数和函数外的变量建立起真正的联系,也就是说函数内部可以直接操作函数外部的变量。例如上例中的Swap2函数
当要改变函数值的时候(比如要交换变量的值等等)用传址调用,其他时候(比如比较两变量值的大小等)用传值调用
函数的嵌套调用和链式访问
嵌套调用:
void new_line()
{
printf("haha\n");
}
void two_line()
{
int i = 0;
for (i = 0; i < 2; i++)
{
new_line();
}
}
int main()
{
two_line();
return 0;
}
链式访问:
把一个函数的返回值作为另一个函数的参数
举一个比较有意思的例子吧,可以猜猜看输出什么,或者去运行一下
int main()
{
printf("%d", printf("%d", printf("%d", 43)));
return 0;
}
因为printf的返回值是输出的字符数量,这里的字符数量包括数字、字母、空格、符号等
函数的声明和定义
函数声明:
- 仅仅是告诉编译器函数叫什么,参数是什么,返回类型是什么,但函数是否真是存在,不重要
- 函数一定是先声明再使用
- 函数的声明一般放在头文件中
函数定义:
交代函数的功能实现
test.h的内容:
函数的声明
int Add(int x, int y);
test.c的内容:
函数的定义
int Add(int x, int y);
#include "test.h"
int Add(int x, int y)
{
return x + y;
}
函数递归
递归的主要思考方式为:把大事化小
递归的两个必要条件:
1.要存在限制条件,当条件满足时递归便不再继续
2.每次递归调用之后要越来越接近这个条件
3.递归层次不能太深,防止栈溢出