• 【C语言】详解函数(下)(庖丁解牛版)


    1. 前言

    详解C语言函数(上)的链接:http://t.csdnimg.cn/EGsfe

    经过对函数的初步了解之后,相信大家已经对C语言标准库里的函数已经有初步的认知了,并且还学会了如何自定义函数。

    在之前我们学过的数据类型中,有整型、字符型、浮点型、布尔类型还有数组。这就会引发一个问题,我们说形参相当于我们给函数的一个可操作的初始变量的值,而在我们之前举的例子中,我都是用整型变量作为形参。那如果我用数组作为形参,又会是怎么样的呢?那么,在本文中就会给大家介绍,数组作为形参的效果

    另外,还会讲解如果函数的函数体里面还有个函数,又会是什么情况。

    那么,让我们一起扬帆起航吧!!!🚢🚢🚢

    2. 数组做函数形参

    在使用函数解决问题时,我们肯定会遇到一种情况:对数组里面的元素进行操作。那这就意味着,我们得把数组作为参数传递给函数,让函数来帮我们处理。那在主函数把参数传递给函数的过程中,会发生什么情况呢?这难免会引起我们对此的一泡浓厚兴趣。

    我们以基本现象来逐步深入问题的本质:
    假如,现在要求你写一个功能:在一个函数将整个数组的内容,全部置为-1,在写一个函数打印数组的内容。

    简单思考一下,不难写出主函数里面的基本框架:

    #include
    int main()
    {
    	int arr[] = {1,2,3,4,5,6,7,8,9,10};
    	int sz = sizeof(arr) / sizeof(arr[0]); //计算数组元素的个数
    	set_arr(); //作用:将数组里面的元素都置为-1。里面的参数有待填写
    	print_arr(); //作用:打印整个数组的元素。里面的参数有待填写
    	return 0;
    }
    

    这⾥的set_arr函数要能够对数组内容进⾏设置,就得把数组作为参数传递给函数,同时函数内部在设置数组每个元素的时候,也得遍历数组,需要知道数组的元素个数。所以我们需要给set_arr传递2个参数,⼀个是数组,另外⼀个是数组的元素个数。仔细分析print_arr也是⼀样的,只有拿到了数组和元素个数,才能遍历打印数组的每个元素。

    #include
    int main()
    {
    	int arr[] = {1,2,3,4,5,6,7,8,9,10};
    	int sz = sizeof(arr) / sizeof(arr[0]); //计算数组元素的个数
    	set_arr(arr,sz); //作用:将数组里面的元素都置为-1。
    	print_arr(arr,sz); //作用:打印整个数组的元素。
    	return 0;
    }
    

    数组作为参数传递给了set_arr和print_arr函数了,那么这两个函数具体如何设计呢?

    在此之前,我们就要得先了解数组传参的几个重要的知识点了(敲黑板,干货来了):

    1. 函数的形式参数要和函数的实参个数匹配
    2. 函数的是参数数组,形参也可以写成数组的形式
    3. 形参如果是一维数组,数组的大小可以省略不写
    4. 形参如果是二维数组,行可以省略,但列不可以省略
    5. 数组传参时,形参是不会创建新的数组的
    6. 形参操作的数组和实参的数组是同一数组

    根据上述的信息,我们可以实现这两个函数了:

    void set_arr(int arr[], int sz)
    {
    	int i = 0;
    	for(i = 0; i < sz; i++)
    	{
    		arr[i] = -1;
    	}
    }
    
    void print_arr(int arr[], int sz)
    {
    	int i = 0;
    	for(i = 0; i< sz; i++)
    	{
    		printf("%d ",arr[i]);
    	}
    	printf("\n");
    }
    

    这样的完成了我们的要求了。相信通过上述的例子和讲解,你已经大概清楚了数组作为函数的参数时,是如何设计自定义函数的形参,以及如何在函数内操作数组的了。

    关于数组作为函数参数进行传参的过程中,还有更多的细节,碍于篇幅的限制,目前只需要了解到这里就已经很不错了。后续我也会写一篇关于数组作为函数参数传参细节的文章,到时候希望大家来捧场。🥺

    3. 函数嵌套调用和链式访问

    有些读者可能对这个概念比较陌生,那我就先讲解一下这个概念,究竟什么时函数的嵌套调用和链式访问?

    3.1 嵌套调用

    嵌套调用就是函数间的相互调用。说白了,就是你中有我,我中有你。
    也可以这么理解,把每个函数想象成一个个乐高零件,正是因为有这么多乐高零件的相互配合、相互成全,才成就出一个巨大且精美的乐高玩具,这也就是函数嵌套调用的精髓所在。

    下面我给一道题目,我们在题目中理解概念:

    题目:假如我们计算某年某月有多少天?

    拿到这个题目时,我们就会想,平年和闰年在2月份有区别。每个月份的天数也有区别。
    根据这个思路,我们就可以设计这两个函数:
    一个是用于判断年份是否位闰年,is_leap_year()
    另一个是用于说明对应月份的对应天数,get_days_of_month()

    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;
    }
    
    

    分析:这⼀段代码,完成了⼀个独⽴的功能。代码中反应了不少的函数调⽤:

    • main 函数调⽤ scanfprintfget_days_of_month
    • get_days_of_month 函数调⽤ is_leap_year

    未来的稍微⼤⼀些代码都是函数之间的嵌套调⽤,但是函数是不能嵌套定义的

    3.2 链式访问

    所谓链式访问就是将⼀个函数的返回值作为另外⼀个函数的参数,像链条⼀样将函数串起来就是函数的链式访问。

    这个其实也很好理解,比如:你现在接收到一个任务,你需要把一个待卡扣绳子的一端,扣在另一根绳子上。这个"卡扣"就是一个函数的返回值,"被扣的那个绳子"就是另一个函数。它们相互配合,共同完成了一个任务。

    下面写一个链式访问(说不定,你在不经意间就已经用过了)的情况:

    #include
    #include
    int main()
    {
    	printf("%d\n", strlen("I love learning C!")); //链式访问
    	return 0;
    }
    

    下面我给大家一段有趣的代码,大家下来可以自己思考一下:

    #include 
    int main()
    {
     	printf("%d", printf("%d", printf("%d", 43)));
     	return 0;
    }
    

    想要知道这个结果,接得先了解printf函数的返回值。这时候,你也许会惊讶,真的假的,我用了这么久的的printf函数,竟然都不知道printf函数还有返回值。

    其实是有的,不信的话,可以去官网找一下这个函数的文档。为了方便研究,我就帮大家找好了。
    printf函数
    printf函数
    看到这里,你就清楚了,printf函数返回的是被成功打印屏幕上的字符的个数。

    上⾯的例⼦中,我们就第⼀个printf打印的是第⼆个printf的返回值,第⼆个printf打印的是第三个printf的返回值。

    第三个printf打印43,在屏幕上打印2个字符,再返回2
    第⼆个printf打印2,在屏幕上打印1个字符,再放回1
    第⼀个printf打印1
    所以屏幕上最终打印:4321

    最后的最后,如果觉得文章写的还不错的话,请多多点赞。你们的认可是我前进和分享知识的动力之一。🙏🙏🙏

  • 相关阅读:
    joblib嵌套式并行运行方式
    UV统计 - HyperLogLog
    Kotlin jetpack compose Tab的渲染 AnimatedVisibility的使用
    Dubbo实战运用Demo、SpringBoot整合Dubbo、Dubbo中超时重试和负载均衡策略
    6、React脚手架
    [数组中等题] LeetCode 969. 煎饼排序
    PostgreSQL 并发控制 -- 锁体系(spinlock,lwlock,regular lock)实现原理
    算法基础-数学知识-高斯消元、求组合数
    蒙牛智慧牧场:最新鲜的牛奶来源于“数字牛”
    [附源码]计算机毕业设计springboot常见Web漏洞对应POC应用系统
  • 原文地址:https://blog.csdn.net/tianxiawushanu/article/details/139485021