距离上一篇博客已经很久啦,在复习C语言的同时码码字写写博客查查资料,顺便复习啦。
目录
前言
接上文,我讲述了函数的概念、库函数、自定义函数以及形参和实参。在本篇中,我将介绍函数的return语句 、数组做函数参数 、嵌套调⽤和链式访问、函数的声明和定义。
一、函数的return语句
在函数的设计时,我们会经常看到return语句。以下是一些需要注意的地方:
- 在main函数中遇到return语句时,整个程序会结束运行,即后面的代码不会执行。在自定义函数中遇到return语句时,函数会返回一个值,然后结束该部分函数的执行。
- return语句后面可以什么都不接,此时适用于返回值为void的情况。
- return语句的后面可以接具体的值(也可以返回地址,地址也是一种值),也可以接一个表达式(若是这种情况,会先计算该表达式的值,然后返回该值)。
- return返回的值和函数返回类型不⼀致,系统会⾃动将返回的值隐式转换为函数的返回类型。
- 如果函数中存在if等分⽀的语句,则要保证每种情况下都有return返回,否则会出现编译错误。
二、数组做函数参数
有时候我们在设置函数的时候,有必要将数组作为参数,来达到所需的目的。可是,我们知道不能通过形参来修改实参,但是如若我们要将一个一维数组的内容全部置为-1,我们传数组是否会有效果呐?答案是可以的。因为传入一维数组名,相当于传入的是该数组首元素的地址,而我们可以通过地址来修改实参,所以结果是可行的。不信可以看看下面的一串代码:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
void set_arr(int arr[], int i)
{
for (int j=0; j < i; j++)
{
arr[j] = -1;
}
}
int main()
{
int arr[10] = { 0 };
int i = 0;
set_arr(arr, 10);
for (i; i < 10; i++)
{
printf("%-4d", arr[i]);
}
return 0;
}
观察上述代码,我们发现在set_arr函数中我们传入了两个参数,一个是数组,一个是整型(为该数组元素的个数)。通过打印,我们发现所有的元素都被置为了-1。这说明,数组可以作为函数元素来传参。
但是,我们如何确定,或者说证实传入的不是整个数组的地址,而是数组首元素的地址呢?下面请出sizeof操作符。
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
int test(int arr[])
{
return sizeof(arr) / sizeof(arr[0]);
}
int main()
{
int arr[3] = { 1,2,3 };
int sz = sizeof(arr) / sizeof(arr[0]);
printf("%d\n", sz);
printf("%d\n", test(arr));
return 0;
}
该代码是在x86的环境下运行的
可以发现,第一行的3是正常的元素个数,而第二行的1表示的只是一个元素,所以说明,数组传参时并没有把整个数组传入进来,数组传参只是传入了数组的首元素的地址。那我们是如何实现对数组所有元素置-1的呢?这将在指针部分为大家介绍。
综上,对于函数的数组传参,我们有以下需要注意的点:
- 函数的形式参数要和函数的实参个数匹配
- 函数的实参是数组,形参也是可以写成数组形式的
- 形参如果是⼀维数组,数组⼤⼩可以省略不写
- 形参如果是⼆维数组,⾏可以省略,但是列不能省略
- 数组传参,形参是不会创建新的数组的
- 形参操作的数组和实参的数组是同⼀个数组
三、函数的嵌套调用
下面请看一串代码:
#define _CRT_SECURE_NO_WARNINGS
#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 get_days_of_month(int y, int m)
{
int days[] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
int day = days[m];
if (is_leap_year(y) && m == 2)
day += 1;
return day;
}
int main()
{
int y = 0;
int m = 0;
scanf("%d %d", &y, &m);
int d = get_days_of_month(y, m);
printf("%d\n", d);
return 0;
}
假设我们计算某年某⽉有多少天,如果要函数实现,可以设计2个函数:
• is_leap_year():根据年份确定是否是闰年
• get_days_of_month():调⽤is_leap_year确定是否是闰年后,再根据⽉计算这个⽉的天数
这⼀段代码,完成了⼀个独⽴的功能。代码中反应了不少的函数调⽤:
• main 函数调⽤ scanf 、 printf 、 get_days_of_month
• get_days_of_month 函数调⽤ is_leap_year
注意:函数是不能嵌套定义的!
四、函数的链式访问
链式访问就是将⼀个函数的返回值作为另外⼀个函数的参数,像链条⼀样将函数串起来。比如:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int main()
{
char st[] = "abcdef";
int sz = strlen(st);
printf("%d", sz);
return 0;
}
这是一个最简单的链式访问,首先调用strlen函数求字符串的长度,然后把长度作为返回值赋值给sz,sz作为printf函数的一个参数。
当然,我们也可以更加直接一点:
#include <stdio.h>
int main()
{
printf("%d\n", strlen("abcdef"));//链式访问
return 0;
}
在这里,我们将strlen函数的返回值直接传给了printf函数。然后接下来,我们来看一串有趣的代码:
#include <stdio.h>
int main()
{
printf("%d", printf("%d", printf("%d", 43)));
return 0;
}
聪明的你猜猜屏幕上会打印什么呢?不知道也没有关系,我们来看一下用了这么久的printf的函数返回值。
哦,我们知道了printf函数的返回值是屏幕上打印的字符的个数。那我们再来分析一下这串代码:
#include <stdio.h>
int main()
{
printf("%d", printf("%d", printf("%d", 43)));
return 0;
}
上⾯的例⼦中,我们就第⼀个printf打印的是第⼆个printf的返回值,第⼆个printf打印的是第三个printf的返回值。
第三个printf打印43,在屏幕上打印2个字符,再返回2
第⼆个printf打印2,在屏幕上打印1个字符,再放回1
第⼀个printf打印1所以屏幕上最终打印:4321
五、函数的定义与声明
#include <stido.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;
}
但是如果将函数的定义放在函数调用之后,VS2022会进行报错。解决方法如下:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int is_leap_year(int y);//函数声明
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;
}
函数的调⽤⼀定要满足先声明后使⽤; 函数的定义也是⼀种特殊的声明,所以如果函数定义放在调⽤之前也是可以的。
总结
今天我们学习了函数的return语句 、数组做函数参数 、嵌套调⽤和链式访问、函数的声明和定义,下节课我们将会学习static和extern。