函数
上次我们学习了在C语言里相当重要的语法-循环,并且通过了一些小例子来帮助理解循环。
今天我们来学习在C语言中同样重要的东西--函数。
那么,函数是什么??
函数是一个可以重复利用的代码块,用于执行一个特定的任务。
函数类型
函数分为两种类型:库函数、自定义函数
库函数:C语言标准库提供,耳熟能详的printf 以及 scanf 都是库函数。库函数在调用时都需要引用头文件。
自定义函数:用户自己编写的函数。
函数组成
函数由函数名和参数,函数体组成。
函数名:该函数的唯一标识符,通过输入函数名来调用此函数
参数:分为形参和实参。形参是形式参数,不会占用内存空间,只有在调用函数时才会执行。实参是一个具体实现,会占用内存空间。
函数体:调用函数时执行的内容。
既然如此,我们为什么要学习函数??函数的作用又是什么呢?
情景导入
我们这里假设一个情景:
这天,A正在写一个小小的程序,但是他发现 “比较两个数,输出最大值” 要经常使用,每次敲都很繁琐,让A敲得有点恼火。那么有没有更好的办法来帮A解决他的苦恼呢?
答案是有,使用函数。前面我们说过,函数是一个可以重复利用的代码块,用于执行一个特定的任务。我们不妨自己写一个函数,然后之后每次需要执行“比较两个数,输出最大值”时都能调用这个函数来更加便捷的实现。
这里我们来帮A实现一下:
#include <stdio.h>
int get_max(int a,int b)
{
if (a>b)
return a;
else
return b;
}
int main()
{
//求两个数字最大值
int a = 0;
int b = 0;
scanf("%d %d", &a, &b);
printf("%d\n", get_max(a,b));
return 0;
}
这里是一个简单的函数,先来看main主体:
我们定义了整形变量a 跟 b,并且初始化为0。然后我们使用了库函数scanf来接收这两个值。
这个时候a跟b就是我们输入的值了。那么,我们可以定义一个get_max函数用来比较a跟b的值(这个函数名是自己取的),然后在get_max函数里放入我们需要对比的两个数 (a, b) (这个是实参)意思是每次调用get_max函数时都比较a,b的值。然后,我们的函数写在main()函数外。
函数格式:返回类型 + 函数名(变量)
{
函数体;
}
我们希望函数的返回类型是int,所以就可以写成 int get_max(int a,int b),这里定义了两个整形变量a , b作为函数的形参,用于接收底下我们调用的函数的那两个实参。然后通过简单的if语句判断结果,将结果return(返回)给底下的函数调用,所以我们输出的就是函数的最大值啦~
以上的例子函数被称为有返回值函数,底下是一个无返回值函数:
#include <stdio.h>
void get_max(int a,int b)
{
if (a>b)
printf("%d\n", a);
else
printf("%d\n", b);
}
int main()
{
//求两个数字最大值
int a = 0;
int b = 0;
scanf("%d %d", &a, &b);
get_max(a,b);
return 0;
}
这个例子是一个无返回值参数,所以我们的函数类型定义为void,底下我们只需要调用函数就可以直接输出函数的结果。
这里来说说两个函数的特点:
有返回值函数:计算一个值并返回,这个值通常会赋给一个变量,或者直接在表达式中使用。这类函数通常用于计算或处理数值,并返回结果
无返回值函数:不返回任何值,通常用于执行某些特定操作。
---------------------------------------------------------------------------------------------------------------------------------
例子
以下我们再来通过两个小例子来理解函数:
例子1:判断素数
输入一个数,判断是否是素数。
int main()
{
int a = 0;
printf("Plase input a num:> ");
scanf("%d", &a);
pd_sushu(a);
return 0;
}
这里我们先定义一个a,a的作用是输入一个数字用于判断,并且调用函数pd_sushu(a)来进行判断。
void pd_sushu(int num)
我们定义一个无返回值函数来判断,这里定义一个形参num来接收实参。
{
int i = 0, ctrl_ = 0;
for(i=2; i<num; i++)
{
if(num % i == 0)
{
ctrl_ = 1;
break;
}
}
if(ctrl_ == 0)
printf("Yes!\n");
else
printf("No...\n");
}
这里就是函数的主体了,我们定义ctrl_来控制变量,至于变量是什么等我慢慢道来。
使用for循环来判断素数,想来想象一下:7要怎么判断是不是素数?
是不是 7/2 看看能不能整除,不行就 7/3 、7/3.......7/6 ,规律就是从2开始,一直到7-1结束
所以我们就从2开始循环,循环到num(想判断的数) -1 就可以了。
if(如果) (num % i == 0)(想判断的数能被整除)
ctrl_ = 1; // 将ctrl_的值变成1(原先是0)
这一步ctrl_的作用是一个开关,如果能被整除就不是素数了,我就关掉这个开关。整除不了的话我就保持开关为0,并且继续循环,再一次判断,如果循环结束了开关都没被关(0)那么我就执行一个判断:
开关为0,意味着没有任何一个数可以整除,就是素数了,我们可以直接打印
开关为1,能被整除那肯定就不是素数,所以输出no。
这题还有个方法,可以更好地利用空间,但局限于篇幅,我这里就不写出来了,感兴趣的可以评论区回复一下。
例子2:二分查找
void search_erfen(int a[], int* n, int* right_)
{
int left = 0;
int right = *right_;
while(left <= right)
{
int mid = (left + right) / 2;
if(a[mid] < *n)
left = mid + 1;
else if(a[mid] > *n)
right = mid - 1;
else
{
printf("Find it, It`s in %d\n", mid);
break;
}
}
if(left > right)
printf("No...\n");
}
int main()
{
int a[10] = {1,2,3,4,5,6,7,8,9,10};
printf("Please enter the number you want to search for:> ");
int right_ = (sizeof(a)/sizeof(a[0]))-1;
int num = 0;
scanf("%d", &num);
search_erfen(a, &num, &right_);
这个是一个简单的二分查找,想来看main()函数主体:
定义了一个数组a[ ] 用来存放十个数据。然后上次我们不是讲过:二分查找需要一个左下标和一个右下标嘛,左下标很简单右下标需要计算,所以我们定义一个right_,计算出数组的元素个数并且赋值给他,然后-1,得到右下标。然后我们定义num用来存储用户需要查找的数据。
search_erfen(a, &num, &right_);
这段函数的意思是将a ,num, right 这些实参传递给函数search_erfen。(这里的三个实参都是以地址的形式存入,那么什么是地址?地址其实是指针,关于指针我们可以后面再说,这里并不影响我们写题) 。并且调用函数
我们来看看函数怎么写的
void search_erfen(int a[], int* n, int* right_)
这里,我们将函数定义为无返回值函数,并且写入三个形参用于接收实参的数据
int left = 0;
int right = *right_;
while(left <= right)
{
int mid = (left + right) / 2;
if(a[mid] < *n)
left = mid + 1;
else if(a[mid] > *n)
right = mid - 1;
else
{
printf("Find it, It`s in %d\n", mid);
break;
}
}
if(left > right)
printf("No...\n");
然后,后面的内容跟我们上次学的“二分查找”就几乎是一模一样了,找到左下标和右下标,然后计算出mid,判断mid与我们需要查找的数的大小,要是大于就是表明我们想查找的数在mid左边,所以我们需要调整right来缩小范围,然后再次判断,如果mid小于数的大小,则表明mid在所需查找的数的右边,就需要缩小左下标来进一步缩小范围,然后反复循环以至于找到我们的数,并且打印。
结语
以上就是c语言的函数初认知了,但是这还只是冰山一角,还有好多的内容诸如函数的定义和声明等等东西,我会放在后面来说。
那么,今天的学习就到这里啦~感谢看到这里的人,再见啦!