【C语言】函数详解第二期,函数的递归

简介: 本文讲解:函数。

 image.gif编辑

人生就像一盒巧克力,你永远不知道下一块是什么滋味—阿甘正传


目录

1、函数的嵌套调用和链式访问

1.1嵌套调用

1.1.1练习

1.2链式访问

1.2.1练习

2、函数的声明与定义

2.1函数的声明

2.2函数的定义

2.3函数的声明和定义

3、函数的递归

3.1递归是什么,为什么要有递归?

3.2递归的两个必要条件

3.2.1练习1

3.2.2练习2

3.3递归与迭代

4、程序的模块化


1、函数的嵌套调用和链式访问

1.1嵌套调用

函数的嵌套调用,我们可以理解为函数之间的嵌套类似于循环里面的嵌套一环套一环,只不过循环是可以在循环体内定义一个循环,函数不能在函数体内定义函数。C语言中函数的定义都是相互平行、相互独立的,也就是说在函数定义时,函数体内不能包含另一个函数的定义,即函数不能嵌套定义,但可以嵌套调用。

1.1.1练习

🤼我们来看一个程序,要求是输入一个数,输出从0到这个数的和:

#include<stdio.h>
int set_num(int x) 
{
  return x + 1;
}
int get_num(int m)
{
  int s = 0;
  for (int i = 0; i < m; i++)
  {
    s=s+set_num(i);
  }
  return s;
}
int main()
{
  int n = 0;
  scanf("%d", &n);
  printf("%d\n", get_num(n));
  return 0;
}

image.gif

输入:4 输出:10


此程序就用到了到了嵌套调用,主函数main里面调用了get_num函数,而get_num函数里面调用set_num函数。返回时set_num函数返回get_num函数返回主函数main。也就是调用时候是顺序,返回是逆序。


1.2链式访问

函数的链式访问是把一个函数的返回值作为另一个函数的参数。

🤼举一例子,求字符串长度:

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[] = "abcdef";
  printf("%d", strlen(arr));
}

image.gif

结果:6


上述程序就是一个简单的链式访问,printf函数里面调用strlen函数的返回值6,从而输出6。


1.2.1练习

🤼‍♀️我们来看一个程序:

#include<stdio.h>
int main()
{
  printf("%d ", printf("%d ", printf("%d ",66)));
  return 0;
}

image.gif

输出:66 3 2


为什么会这样呢,最里面的 printf输出66 ,第二个printf输出3 第一printf输出2。注意格式符%d后面有个空格。printf函数返回值类型是整形。因此每次返回的都是前一个字符个数。

image.gif编辑


2、函数的声明与定义

2.1函数的声明

    • 函数的声明一般出现在函数的使用之前。要满足先声明后使用。
    • 函数的声明一般要放在头文件中的

    函数的声明就是我们要向编译器写出函数的返回类型是声明,函数名是什么,参数是什么。具体函数体是否存在,函数的声明解决不了,我们要看函数的定义。下图就是函数的声明,表明了函数的返回类型,函数名,参数。

    image.gif编辑


    2.2函数的定义

    函数的定义就是函数体里面的内容,也就是函数的实现功能。下图展示了函数的声明、定义的位置。

    image.gif编辑


    2.3函数的声明和定义

    函数在被调用前必须要声明自己的返回类型,函数名,参数。

    #include<stdio.h>
    int main()
    {
      int a = 0;
      int b = 0;
      scanf("%d %d", &a, &b);
      printf("%d\n",Sum(a, b));
      return 0;
    }
    int Sum(int x, int y)
    {
      return x + y;
    }

    image.gif

    输入:1 2 输出:3


    上述程序输入输出都没什么问题,但是没有在主函数上方声明。导致最后报警告,因此我们应该在主函数上方声明一下。image.gif编辑


    image.gif编辑


    3、函数的递归

    3.1递归是什么,为什么要有递归?

    首先我们要知道递归的过程为递推与回归。递归做为一种算法,它是一个过程或函数在定义后在函数本身中自己调用自己的一种算法。它通常把一个大型复杂问题,层层化为原问题相似的小规模问题来求解这讲究递归策略。优点是只需要少量的代码解决多次的重复计算。


    🤼‍♀️我们来看一程序:

    #include<stdio.h>
    int main()
    {
      printf("Hello World\n");
      main();
      return 0;
    }

    image.gif

    结果:一直打印Hello World


    主函数main()中调用main()。这就是一个递归的思想,自己调用自己。只不过该程序没有终止条件,导致程序最后栈溢出,只得编译器自行终止。


    3.2递归的两个必要条件

    首先我们要知道递归过程为递推然后回归,那么递归过程中必须满足两个条件:

      • 递归中必须存在限制条件,当达到这个限制条件时,递归将不再继续。
      • 每次递归调用后会越来越接近这个限制条件。

      3.2.1练习1

      🤼接受一个无符号整形数,按照顺序打印它的每一位,例如输入:1234,输出1 2 3 4。

      #include<stdio.h>
      void Print(unsigned int n)
      {
        if (n > 9)
        {
          Print(n / 10);
        }
        printf("%d ", n % 10);
      }
      int main()
      {
        unsigned int n = 0;
        scanf("%u", &n);
        Print(n);
        return 0;
      }

      image.gif

      输入:1234

      输出:1 2 3 4


      上面我们说到了,递归分为两步递推和回归,所以此程序分为两步

      第一步:递推,我们每次递推(调用)后的下一次递推,if语句里面的n越来越接近终止条件。第一次递推if语句中n=123第二次递推if语句中n=12第三次递推if语句中n=1,执行printf语句并输出1 。当n=1时终止递推,此时执行printf语句输出1 。


      第二步:回归,每次回归(回调)时也是要调用Print函数的只不过每次判断的时候都为假,每次的回归n始终为1,原n还是原来的值。第一次回归判断n=1,原n=12,执行printf语句并输出2 第二次回归判断n=1,原n=123,执行printf语句并输出3 第三次回归判断n=1,原n=1234,执行printf语句并输出4 。我们可以这样理解,回归的每一次判断都为假,但原来n的值还在内存中存放着,所以每次输出的都是原n的值%10。


      最后:屏幕上打印1 2 3 4

      image.gif编辑

      递推:递推前n=1234,第一次递推n=123,第二次递推n=12,第三次递推n=1,结束递推,执行printf语句,输出1 。


      回归:第一次回归n=1,原n=12,if判断为假输出 2。第二次回归n=1,原n=123,if判断为假输出 3。第三次回归n=1,原n=1234,if判断为假输出4 。


      最后:可能有小伙伴说那你Print函数怎么不返回主函数呢?因为Print函数的返回类型为void所以并不返回主函数,我们只是在函数里面调用函数(递归)。可以理解为每一次的调用都是形参,所以每一次调用前的n都还是原来的n,只不过最后一次返回的n是1导致返回给之前的每一次if判断都为假。如果您函数的返回与不返回或者实参与形参的概念还不太懂,可以看看我上一期的函数讲解😊


      unsigned:无符号位格式符%u


      3.2.2练习2

      题目:编写函数不允许创建临时变量,求字符串的长度

      🤼‍♀️我们先来看允许使用教临时变量的情况:

      #include<stdio.h>
      int Str(char* str)
      {
        int count = 0;
        while (*str != '、0')
        {
          str++;
          count++;
        }
        return count;
      }
      int main()
      {
        char arr[] = "abcdef";
        printf("%d",Str(arr));
        return 0;
      }

      image.gif

      结果:6


      上述程序中,主函数main定义一个名为arr的字符串类型数组, Str函数创建一个字符型指针变量str来接受这个arr数组。那么str默认指向的是arr数组的首地址也就是第一个元素的a的地址。并且arr数组里面的每个元素的地址是连续的。知道这个原理后,我们来解答Str函数内部原理。


      第一次:对str解引用if判断*str!='/0',str++,count++。此时str指向arr第二个元素地址,count=1。

      第二次:再次对str解引用if判断*str!='/0',str++,count++。此时str指向arr第三个元素地址,count=2。

      依此类推(省略四次)

      最后一次:对str解引用if判断*str='/0'结束while循环,此时count=6。返回6给实参,并输出6。

      image.gif编辑


      🤼不允许创建临时变量情况下 :

      #include<stdio.h>
      int Str(char* str)
      {
        if (*str != '\0')
        return 1 + Str(str+1);
        return 0;
      }
      int main()
      {
        char arr[] = "abc";
        printf("%d\n",Str(arr));
        return 0;
      }

      image.gif

      输出:3


      那么不采用临时变量的情况下,我们就用递归来解决这个程序。也是与上个程序一样,主函数里面定义一个arr数组。Str函数定义一个指针字符变量str来接受,也是指向该数组的第一个元素的地址。并且arr数组里面各个元素的地址是连续的,知道了这些原理后。我们来解答递归的过程。首先我们看一组图片,然后结合下方解析理解。

      image.gif编辑

      第一次递推:str指向的地址为b的地址,此时1+Str(str+1),传过去的参数为bc\0的地址

      第二次递推:str指向的地址为c的地址,此时1+1+Str(str+1),传过去的参数为c\0的地址

      第三次递推:str指向的地址为\0的地址,此时1+1+1+Str(str+1),传过去的参数为\0的地址,结束程序;

      第一次回归为:0+1;

      第二次回归为:0+1+1;

      第三次回归为:0+1+1+1;

      Str函数返回给主函数的值为:3,最后输出3。


      image.gif编辑

      相信大家通过上面两个练习懂得了递归的原理,我们来看下一节


      3.3递归与迭代

      迭代就是循环,上面我们说到了,递归就是把一个大的问题化成多个小的问题来解决。但当一个问题变得特别大的时候,递归真的就非常的适合我们去使用吗?我们来看一个程序:求第n个斐波那契数(不考虑溢出)

      #include<stdio.h>
      int Fib(int n)
      {
        if (n <= 2)
          return 1;
        else
          return Fib(n - 1) + Fib(n - 2);
      }
      int main()
      {
        int n = 0;
        scanf("%d", &n);
        printf("%d", Fib(n));
        return 0;
      }

      image.gif

      输入:10 输出:55

      输入:30 输出:832040


      斐波那契数列:1 1 2 3 5 8 13 21 34 55 89 ....... ,当我们输入的项数很小的时候,程序能立马运行出来,例如输入10和30。但如果输入50程序此时就不能立马运算出来,就如下图一样。第10项都需要运算这么多次,更别说第50项了。所以递归在问题过大的时候并不适合应用。

      image.gif编辑

      那递归不方便我们采用什么方法来算呢?我们可以采用迭代的方式来设计程序,迭代就是循环。


      🤼‍♀️用迭代,求第n项斐波那契数(不考虑溢出)

      #include<stdio.h>
      int Fib(int n)
      {
        int a = 1;
        int b = 1;
        int c = 0;
        while (n>2)
        {
          c = a + b;
          a = b;
          b = c;
          n--;
        }
        return c;
      }
      int main()
      {
        int n = 0;
        scanf("%d", &n);
        printf("%d\n", Fib(n));
      }

      image.gif

      输入:50 输出:一个随机数


      因为第50项的数太大了,因此导致程序随机生成一个数,但我想表达的是同样一个问题用递归和迭代来做,效率是不同的。同一个类型程序我用递归写出的结果很慢而我用迭代来写出的结果非常快,这就是迭代与递归效率的不同。


      4、程序的模块化

      为了方便于开发,在我们程序的设计时,为了某一功能的实现我们程序员开发一段程序和子程序(包括函数),如果我们把这些程序全写在main函数里面,那将非常的不利于我们调试,也不方便他人使用,因此我们要模块化程序。模块化的好处是:

        • 实现方便利于调试查错,提高代码的维护性
        • 多人使用,提供代码的共享性
        • 课对程序进行封装,提高代码的隐蔽性

        🤼那么有一程序,求两数之间最大的那位数:

        #include<stdio.h>
        int Max(int x, int y)
        {
          return x > y ? x : y;
        }
        int main()
        {
          int a = 0;
          int b = 0;
          scanf("%d %d", &a, &b);
          printf("%d\n", Max(a, b));
          return 0;
        }

        image.gif

        输入:5 3 输出:5


        正常来讲,这是我们写出的程序 ,这是比较简单的程序几行代码就能搞定。但是当我们遇见相当复杂的程序,代码非常的长。主函数里面还要有其他的代码,这时候我们可以模块化程序。还是拿上面程序举例子,我们可以这样做:创建两个名为add.ctest.c的源文件,创建一个名为add.h的头文件。我们把Max函数的声明写在add.h里面,Max函数的定义写在add.c源文件里面,主函数main写在test.c里面。这样我就可以从test.c里面调用add.h,类似于库函数的头文件。如何操作见下图:

        image.gif编辑


        我们可以看到,test.c函数里面引入add.h头文件。这样我们的程序就模块化了,通过这一个小例子,我们可以了解到程序的模块化基本概念,后期博客中我会深入讲解模块化操作。


        本次博客就到这里结束了,感谢大家的阅读。

        image.gif编辑

        相关文章
        |
        1月前
        |
        C语言 C++
        C语言 之 内存函数
        C语言 之 内存函数
        34 3
        |
        11天前
        |
        C语言
        c语言调用的函数的声明
        被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
        27 6
        |
        1月前
        |
        存储 缓存 C语言
        【c语言】简单的算术操作符、输入输出函数
        本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
        36 10
        |
        24天前
        |
        存储 算法 程序员
        C语言:库函数
        C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
        |
        30天前
        |
        机器学习/深度学习 C语言
        【c语言】一篇文章搞懂函数递归
        本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
        60 7
        |
        30天前
        |
        存储 编译器 程序员
        【c语言】函数
        本文介绍了C语言中函数的基本概念,包括库函数和自定义函数的定义、使用及示例。库函数如`printf`和`scanf`,通过包含相应的头文件即可使用。自定义函数需指定返回类型、函数名、形式参数等。文中还探讨了函数的调用、形参与实参的区别、return语句的用法、函数嵌套调用、链式访问以及static关键字对变量和函数的影响,强调了static如何改变变量的生命周期和作用域,以及函数的可见性。
        30 4
        |
        1月前
        |
        存储 编译器 C语言
        C语言函数的定义与函数的声明的区别
        C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。
        |
        27天前
        |
        存储 C语言
        【c语言】字符串函数和内存函数
        本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
        23 0
        |
        27天前
        |
        C语言
        【c语言】qsort函数及泛型冒泡排序的模拟实现
        本文介绍了C语言中的`qsort`函数及其背后的回调函数概念。`qsort`函数用于对任意类型的数据进行排序,其核心在于通过函数指针调用用户自定义的比较函数。文章还详细讲解了如何实现一个泛型冒泡排序,包括比较函数、交换函数和排序函数的编写,并展示了完整的代码示例。最后,通过实际运行验证了排序的正确性,展示了泛型编程的优势。
        20 0
        |
        30天前
        |
        算法 C语言
        factorial函数c语言
        C语言中实现阶乘函数提供了直接循环和递归两种思路,各有优劣。循环实现更适用于大规模数值,避免了栈溢出风险;而递归实现则在代码简洁度上占优,但需警惕深度递归带来的潜在问题。在实际开发中,根据具体需求与环境选择合适的实现方式至关重要。
        27 0