函数(function)的概念,有些翻译为:子程序,子程序这种翻译更加准确⼀些。C语言中的函数就是一个完成某项特定的任务的一小段代码。C语⾔的程序其实是由⽆数个小的函数组合而成的,也可以说:一个大的计算任务可以分解成若干个较小的函数(对应较小的任务)完成。
我们前面内容中学到的 printf 、 scanf 都是库函数,库函数也是函数,不过这些函数已经是现成的,我们只要学会就能直接使用了。有了库函数,⼀些常见的功能就不需要程序员自己实现了,⼀定程度提升了效率。同时库函数的质量和执行效率上都更有保证。但是要使用库函数的话,头文件是必不可少的。
库函数相关头文件:C 标准库头文件 - cppreference.com
C/C++官方的链接:C 标准库头文件 - cppreference.com cplusplus.com:C library - C++ Reference
#include<stdio.h>
#include<math.h>
int main()
{
double a = 16;
double ret = sqrt(a);
printf("ret=%lf", ret);
return 0;
}
这里我们就用到了一个头文件(math.h)开平方,sqrt(a)的意思是对a变量存储的值开平方。这里一定要表明头文件math.h,这样才能调用math库函数
结果:
ret=4.000000
以上就是常见的库函数,如果对于哪里的函数不了解,可以根据我上面的网站自行学习。
自定义函数和库函数是一样的,本质定义差不多
定义:
ret_type fun_name(形式参数) { }
例如:写一个减法函数
#include<stdio.h>
int Add(int x, int y)
{
return x + y;
}
int main()
{
int a = 15;
int b = 10;
int ret = Add(a, b);//调用加法函数,将返回值放入ret中
printf("ret=%d", ret);
return 0;
}
结果
ret=5;
举例:写两个交换的函数,一个传实参,一个传形参,判断目的是否达到
#include <stdio.h>
//实现成函数,但是不能完成任务
void Swap1(int x, int y)
{
int tmp = 0;
tmp = x;
x = y;
y = tmp;
}
//正确的版本
void Swap2(int *px, int *py)
{
int tmp = 0;
tmp = *px;
*px = *py;
*py = tmp;
}
int main()
{
int num1 = 1;
int num2 = 2;
Swap1(num1, num2);
printf("Swap1::num1 = %d num2 = %d\n", num1, num2);
Swap2(&num1, &num2);
printf("Swap2::num1 = %d num2 = %d\n", num1, num2);
return 0;
}
我们用F11进入调试界面,然后打开见识界面,观察x,y形参地址和num1、num2的地址。观察他们的地址是否相同。
这里可以看到 Swap1 函数在调用的时候,x,y拥有自己的空间,同时拥有了和实参一模一样的内容。 所以我们可以简单的认为:
形参实例化之后其实相当于实参的一份临时拷贝
在使用函数解决问题的时候,难免会将数组作为参数传递给函数,在函数内部对数组进⾏操作。 ⽐如:写⼀个函数打印整型数组的内容。
#include<stdio.h>
void Print(int arr2[], int sz)
{
int i = 0;
for (i = 0;i < sz;i++)
{
printf("%d ", arr2[i]);
}
}
int main()
{
int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
int sz = sizeof(arr1) / sizeof(arr1[0]);//求数组元素的个数
Print(arr1, sz);
return 0;
}
void new_line()//被调用函数
{
printf("hehe\n");
}
void three_line()
{
int i = 0;
for(i=0; i<3; i++)
{
new_line();//嵌套调用
}
}
在一个函数编写的过程中调用非自身函数的函数叫做嵌套调用,在实际编写的过程的中,可以为我们省去非常多的麻烦,没必要实现相应的功能再重写编写一边代码。
//1.strlen返回一个字符串的长度
printf("%d\n", strlen("abcdef"));//链式访问
//2.打印长度6
return 0;
这就是一个非常典型的链式访问,用strlen函数的返回值作为printf的参数,这就叫做链式访问
这也是非常核心的问题
我们来看一个非常有趣的问题,这也是某个大厂的面试题:
#include <stdio.h>
int main()
{
printf("%d", printf("%d", printf("%d", 43)));
return 0;
}
结果是4321
为什么?
既然是链式访问,就一定要用函数的返回值,问题就来了,我们平常用的是printf的实际功能,却从来关心过printf的返回值。
print有返回值,返回的是字符的个数
⼀般我们在使用函数的时候,直接将函数放入一个文件中。 比如:我们要写⼀个函数判断⼀年是否是闰年。
#include<stdio.h>
int is_leap_year(int y)//函数的定义,判断一年是不是闰年
{
if (((y % 4 == 0) && (y % 100 != 0)) || (y % 400 == 0))
return 1;
else
return 0;
}
int main()
{
int y = 0;
scanf("%d", &y);
int r = is_leap_year(y);//函数的调用
if (r == 1)
printf("闰年\n");
else
printf("⾮闰年\n");
return 0;
}
我们将is_leap_year函数放到主函数的后面,运行调试:
#include<stdio.h>
int main()
{
int y = 0;
scanf("%d", &y);
int r = is_leap_year(y);
if (r == 1)
printf("闰年\n");
else
printf("⾮闰年\n");
return 0;
}
int is_leap_year(int y)
{
if (((y % 4 == 0) && (y % 100 != 0)) || (y % 400 == 0))
return 1;
else
return 0;
}
为什么在放在后面就无法识别,其实这是由c语言的顺序结构来决定的,从上到下依次运行,如果我们上面没有函数的影子,编译器就认为并没有定义和编写这个自定义函数,所以我们使用函数的时候我们一般都要先声明后使用(函数编写在主函数后的)
感谢各位观看至此,如果有什么不足的地方,欢迎大家私信,我深知自己见识浅薄,对于前方险阻,我们一起进步