C语言-函数2

简介: C语言-函数2

函数的声明和定义

函数声明:

告诉编译器有一个函数叫什么,参数是什么,返回类型是什么。但是具体是不是存在,函数声明决定不了。

函数的声明一般出现在函数的使用之前。要满足先声明后使用。

函数的声明一般要放在头文件中的。

#include <stdio.h>
int Add(int x, int y);//提前声明
int main()
{}
int Add(int x, int y) //定义
{}

函数定义:

函数的定义是指函数的具体实现,交待函数的功能实现。

未来在工程中代码是比较多的 函数在add.h(头文件)中声明

                                                           在add.c(源文件)中定义

                                                           在test.c(源文件)中实现

在test.c前有写 #include "add.h" //自己写的一般用""就可以了 //库函数的写成<>

从编译器到链接器的过程是由 VS2022-集成开发环境实现的

为什么要将代码拆成.c/.h的文件?

1. 多人协作

最后将代码整合

2. 代码保护

可以把代码变成静态库 可以把代码卖给别人而不暴露代码(.lib 和 .h 传给别人)

函数递归

什么是递归?

程序调用自身的编程技巧称为递归( recursion)。

递归做为一种算法在程序设计语言中广泛应用。

一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解, 递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算,大大地减少了程序的代码量。

递归的主要思考方式在于:把大事化小

递推+回归

最后栈溢出 // 程序错误

递归的两个必要条件

存在限制条件,当满足这个限制条件的时候,递归便不再继续

每次递归调用之后越来越接近这个限制条件

例子

接受一个整型值(无符号),按照顺序打印它的每一位。

例如:输入:1234,输出 1 2 3 4.

#include <stdio.h>
int main(){
  //1234
  //1234%10 = 4
  //1234/10 = 123
  //123%10 = 3
  int num = 0;
  scanf("%d",&num);
  //非递归的方法
  while(num){
    printf("%d ", num%10);
    num = num / 10;
  }
  //递归的方法
  //Print能将参数num的每一位打印在屏幕上
  Print(num);
  return 0;
}
void Print(int n){
  if(n > 9){
    Print(n/10);
  }
  printf("%d",n%10);
}

函数之所以能实现递归(调用)

都是因为,函数在调用的时候会维护一个函数栈帧

调用的时候创建 结束的时候销毁

如图 函数结束就直接销毁(图中已经销毁了 n = 12 的Print 栈区)

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

不允许创建临时变量

利用递归来实现

#include <stdio.h>
//模拟实现strlen 
int Strlen(const char*str)//从第一位的地址开始求
{
    if(*str == '\0')
        return 0;
    else
        return 1+Strlen(str+1);//从下一位的地址继续求
}
int main() {
    char *p = "abcdef";
    int len = Strlen(p);
    printf("%d\n", len);
    return 0;
}

库函数strlen的返回值是size_t类型

size_t 是无符号整型 %zd 为了sizeof 设计的

函数调用数组的时候传递的是数组首字母的地址

strlen 统计的是\0 之前的字符个数

递归与迭代

例子

求n的阶乘。(不考虑溢出)
int factorial(int n)
{
    if(n <= 1)
        return 1;
    else
        return n * factorial(n-1);
}
求第n个斐波那契数。(不考虑溢出)

1 1 2 3 5 8 13 21 34 55 89

int fib(int n)
{
    if (n <= 2)
        return 1;
    else
        return fib(n - 1) + fib(n - 2);
}

但是我们发现有问题

在使用 fib 这个函数的时候如果我们要计算第50个斐波那契数字的时候特别耗费时间。

使用函数求10000的阶乘(不考虑结果的正确性),程序会崩溃。

为什么呢?

我们发现 fib 函数在调用的过程中很多计算其实在一直重复。 如果我们把代码修改一下:

int count = 0;//全局变量 int fib(int n)
{
    if(n == 3)
        count++;
    if (n <= 2)
        return 1;
    else
        return fib(n - 1) + fib(n - 2);
}

最后我们输出看看count,是一个很大很大的值。 那我们如何改进呢?

在调试函数的时候,如果你的参数比较大,那就会报错: stack overflow (栈溢出)

这样的信息。

系统分配给程序的栈空间是有限的,但是如果出现了死循环,或者(死递归),这样有可能导致一 直开辟栈空间,最终产生栈空间耗尽的情况,这样的现象我们称为栈溢出

那如何解决上述的问题:

  1. 将递归改写成非递归。
  2. 使用static对象替代nonstatic局部对象。在递归函数设计中,可以使用static对象替代nonstatic 局部对象(即栈对象),这不仅可以减少每次递归调用和返回时产生和释放对象的开销,而且存递归调用的中间状态,并且可以各个调用层所访问。 比如,下面代码就采用了,非递归的方式来实现:
//求n的阶乘
int factorial(int n) {
}
int result = 1;
while (n > 1)
{
result *= n ;
n -= 1; }
return result;
//求第n个斐波那契数 int fib(int n)
{
     int result;
     int pre_result;
     int next_older_result;
     result = pre_result = 1;
     while (n > 2) {
        n -= 1;
        next_older_result = pre_result;
        pre_result = result;
        result = pre_result + next_older_result;
     }
     return result;
}
相关文章
|
28天前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
663 0
|
3月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
252 15
|
9月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
399 23
|
8月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
220 1
一文彻底搞清楚C语言的函数
|
9月前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
322 15
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
|
9月前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
165 24
|
9月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
390 16
|
9月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
218 3
|
9月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
184 2
|
9月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
156 1