【C语言】函数的声明_函数定义_函数调用_函数递归 [函数的基本使用]

news2024/11/17 17:24:40

文章目录

  • 前言
  • 1.函数是什么?
  • 2.C语言中函数的分类
    • 2.1 库函数
    • 2.2 自定义函数
  • 3.函数的参数
    • 3.1 实际参数(实参):
    • 3.2 形式参数(形参):
  • 4.函数的调用
    • 4.1 传值调用
    • 4.2 传址调用
    • 4.3 练习
  • 5.函数的嵌套调用和链式访问
    • 5.1 嵌套调用
    • 5.2 链式访问
  • 6.函数的声明和定义
    • 6.1 函数声明:
    • 6.2 函数定义:
  • 7.函数递归
    • 7.1 什么是递归?
    • 7.2 递归的两个必要条件
      • 7.2.1 练习1
      • 7.2.2 练习2
    • 7.3 递归与迭代
      • 7.3.1 练习3
      • 7.3.4 练习4

前言

在前面我们已经学习了初识C语言部分和分支循环语句,接下来我们开始函数部分的学习,函数在我们编程的时候是经常使用的,虽然之前接触的不是很多,但是其实函数的学习并不是很困难,只要稳扎稳打,还是很容易学的通透的,像三子棋、扫雷游戏中都用到了函数,那么函数是什么呢?接下来我们开始函数的学习。

1.函数是什么?

数学中我们经常见到函数的概念,但是你真的了解函数的概念吗?
维基百科中对函数的定义:子程序
在计算机科学中,子程序是一个大型程序中的某部分代码,由一个或多个语句块组成。它负责完成某项特定任务,而且相较于其他代码,具备相对的独立性。
一般会有输入参数并有返回值,提供对过程的封装和细节的隐藏。这些代码通常被集成为软件库。

2.C语言中函数的分类

1.库函数
2.自定义函数

2.1 库函数

为什么会有库函数?
1.我们知道在我们学习C语言编程的时候,总是在一个代码编完之后迫不及待的想要知道结构,想把这个结果打印到我们的屏幕上看一看。这个时候我会频繁的使用一个打印函数(printf)。
现在我们试想一下,如果每个程序员都有一个自己的打印函数,那么会有什么影响?

1.代码冗余
2.开发效率低
3.不标准

2.在编程的过程中我们会频繁的使用字符串拷贝函数(strcpy)
3.在编程时我们也会计算,计算n的k次方这样的运算(pow)
像上面我们描述的基础功能,它们不是业务性的代码。我们在开发过程中每个程序员都可能用得到,为了支持可移植和提高程序的效率,所以C语言中提供了一系列类似的库函数,方便程序员进行软件开发。
那我么怎么学习库函数呢?
www.cplusplus.com
在这里插入图片描述
简单的总结,C语言常用的库函数有:
IQ函数
字符串操作函数
字符操作函数
内存操作函数
时间/日期函数
数学函数
其他库函数

接下来我们根据文档来学习一个库函数
strcpy
在这里插入图片描述
因为有些文档是全英文的,所以我们要锻炼读英文文档的能力,下面通过一段代码来演示strcpy函数:

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "XXXXXXXXXXXX";
	strcpy(arr2, arr1);
	printf("%s\n", arr2);
	return 0;
}

文档中说原字符串的\0也被拷贝进去
现在我们通过调试来观察\0是否被拷贝:
在这里插入图片描述
注意:
库函数必须知道的一个秘密就是:使用库函数,必须包含#include对用的头文件。
这里对照文档学习上面几个库函数,目的就是掌握库函数的使用方法。

2.2 自定义函数

如果库函数能干所有的事情,那还要程序员干什么?
所以更加重要的是自定义函数。
自定义函数和库函数一样,都有函数名,返回类型和函数参数。
不一样的是这些都是我们自己来设计,这给程序员一个很大的发挥空间。
函数的组成:

ret_type fun_name(para, *)
{
	statement;//语句项
}
ret_type 返回类型
fun_name 函数名
para1    函数参数

我们举一个例子:使用函数求两个数的最大值

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

再举一个例子:写一个函数交换两个整形变量的内容

#include <stdio.h>
void swap(int* x, int* y)
{
	int tmp = *x;
	*x = *y;
	*y = tmp;

}
int main()
{
	int a = 10;
	int b = 20;
	printf("交换前a=%d,b=%d\n", a, b);
	swap(&a, &b);
	printf("交换后a=%d,b=%d\n", a, b);
	return 0;
}

3.函数的参数

3.1 实际参数(实参):

真是传给函数的参数,叫做实参。
实参可以是:常量、变量、表达式、函数等。
无论实参是何种类型的量,在进行函数调用时,它们必须有确定的值,以便把这些值传送给形参。

3.2 形式参数(形参):

形式参数是指函数名后面括号中的变量,因为形式参数只有在函数被调用的过程中才实例化(分配内存单元),所以叫形式参数。形式参数当函数调用完成之后自动销毁。因为形式参数只有在函数中有效。
总结:形参实例化之后其实相当于实参的一份临时拷贝。

4.函数的调用

4.1 传值调用

函数的形参和实参分别占用不同的内存块,对形参的修改不会影响实参。

4.2 传址调用

传址调用是把函数外部创建变量的内存地址传递给函数参数的一种函数调用方式。
这种传参方式可以让函数和函数外面的变量建立真正的联系,也就是函数内部可以直接操作函数外部的变量。

4.3 练习

1.写一个函数判断一个数是不是素数。

#include <stdio.h>
void is_prime(int n)
{
	int i = 0;
	for (i = 2; i < n; i++)
	{
		if (n % i == 0)
		{
			break;
		}
		
	}
	if (i == n)
		printf("%d\n", n);
}
int main()
{
	int num = 0;
	scanf("%d", &num);
	is_prime(num);
	return 0;
}

2.写一个函数判断一年是不是闰年

闰年:1.能被4整除但不能被100整除 2.能被400整除

#include <stdio.h>
void is_leap_year(int n)
{
	if (((n % 4 == 0) && (n % 100 != 0)) || (n % 400 == 0))
	{
		printf("%d ", n);
	}
}
int main()
{
	int year = 0;
	scanf("%d", &year);
	is_leap_year(year);
	return 0;
}

3.写一个函数,实现二分查找

#include <stdio.h>
//二分查找的实现
void find(int arr[], int sz)
{
	int k = 7;
	int left = 0;
	int right = sz - 1;
	while (left <= right)
	{
		int mid = left + ((right - left) / 2);
		if (k > arr[mid])
		{
			left = mid;
		}
		else if (k < arr[mid])
		{
			right = mid;
		}
		else
		{
			printf("找到了,下标是%d\n", mid);
			break;
		}
	}
	if (left > right)
	{
		printf("找不到\n");
	}
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	find(arr, sz);
	return 0;
}

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

函数和函数之间可以根据实际的需求进行组合,也就是互相调用的。

5.1 嵌套调用

#include <stdio.h>
void new_line()
{
printf("hehe\n");
}
void three_line()
{
int i = 0;
for(i=0; i<3; i++)
{
new_line();
}
}
int main()
{
three_line();
return 0;
}

函数可以嵌套调用,但是不能嵌套定义。

5.2 链式访问

把一个函数的返回值作为另一个函数的参数。

#include <stdio.h>
int main()
{
	
	printf("%d", printf("%d", printf("%d", 43)));
	//屏幕上打印的数字:4 3 2 1
	return 0;
}

6.函数的声明和定义

6.1 函数声明:

1.告诉编译器有一个函数叫什么,参数是什么,返回类型是什么。但是具体是不是存在,函数声明决定不了。
2.函数的声明一般出现在函数的使用之前。要满足先声明后使用
3.函数的声明一般放在头文件中。

6.2 函数定义:

函数定义是函数的具体实现,交代函数的功能实现。
test.h的内容:放置函数声明

//函数的声明
int Add(int x, int y);

test.c的内容:函数定义

int Add(int x, int y)
{
return x+y;
}

7.函数递归

7.1 什么是递归?

程序调用自身的编程技巧称之为递归。
递归做为一种算法在程序设计语言中广泛使用。一个过程或函数在定义或说明中直接或间接调用自身的一种方法,它通常把一个大型复杂的问题转化为一个与原问题相似的规模较小的问题来求解。
递归策略:只需要少量的程序就可以描述解题过程所需要的程序计算,大大减少了程序的代码量。
递归主要思考方式在于:把大事化小

7.2 递归的两个必要条件

1.存在限制条件,但满足这个限制条件的时候,递归便不再继续。
2.每次递归调用之后越来越接这个限制条件。

7.2.1 练习1

接收一个整形值,按顺序打印每一位
例如:1234 打印:1 2 3 4

void print(int num)
{
	if (num > 9)
	{
		print(num / 10);
	}
	printf("%d ", num % 10);
}
int main()
{
	unsigned int num = 0;
	scanf("%d", &num);
	print(num);
	return 0;
}

7.2.2 练习2

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

int Strlen(char* arr)
{
	if ('\0' == *arr)
		return 0;
	else
		return 1 + Strlen(arr + 1);
	
}
int main()
{
	char arr[] = "abcdef";
	int len = Strlen(arr);
	printf("%d\n", len);
	return 0;
}

7.3 递归与迭代

7.3.1 练习3

求n的阶乘(递归)

int fac(int n)
{
	if (n <= 1)
		return 1;
	else
		return n * fac(n - 1);
}
int main()
{
	int n = 0;
	scanf("%d", &n);
	int z = fac(n);
	printf("%d\n", z);
	return 0;
}

非递归

int fac(int n)
{
	int i = 0;
	int ret = 1;

	for (i = 1; i <= n; i++)
	{
		ret *= i;

	}
	return ret;
}
int main()
{
	int n = 0;
	scanf("%d", &n);
	int z = fac(n);
	printf("%d\n", z);
	return 0;
}

7.3.4 练习4

斐波那契数列:
斐波那契数列(Fibonacci sequence),又称黄金分割数列、因数学家列昂纳多·斐波那契(Leonardoda Fibonacci)以兔子繁殖为例子而引入,故又称为“兔子数列”,指的是这样一个数列:1、1、2、3、5、8、13、21、34、……

规律是:这个数列从第3项开始,每一项都等于前两项之和。
斐波那契数列(递归)

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);
	int ret = fib(n);
	printf("%d\n", ret);
	return 0;
}

非递归

int fib(int n)
{
	int a = 1;
	int b = 1;
	int c = 1;
	while (n > 2)
	{
		c = a + b;
		a = b;
		b = c;
		n--;
	}
	return c;
}
int main()
{
	int n = 0;
	scanf("%d", &n);
	int ret = fib(n);
	printf("%d\n", ret);
	return 0;
}

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

如何解决死递归问题:
1.将递归改写成非递归。
2.使用static对象替代nonstatic局部对象。在函数设计中,可以使用static对象代替nonstatic局部对象(栈对象),这不仅可以减少每次递归调用和返回时产生和释放nonstatic对象的开销,而且static对象还可以保存递归调用的中间状态,并且可为各个调用层所访问。
提示:
1.许多问题是以递归的形式进行解释的,这只是因为它比非递归的形式更加清晰。
2.但是这些问题的迭代实现往往比递归实现的效率更高,虽然代码的可读性稍微差些。
3.当一个问题相当复杂,难以用迭代实现时,此时递归的简洁性便可以补偿它所带来的运行开销。

此外,函数递归还有两个个经典题目(自主研究):
1.汉诺塔问题
2.青蛙跳台阶问题
函数部分的内容到这里就结束了,觉得文章内容还不错的话就点个关注吧,以后我还是会继续更新C语言、C++、数据结构、Linux系统编程+网络编程等内容。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/107535.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

羊没羊,好像也没那么重要了!

疫情管控刚一放开&#xff0c;我就一直在想&#xff0c;如何降低羊&#x1f411;的概率和影响。​由于家里老人身体不太好&#xff0c;孩子年龄又太小&#xff0c;加上只有我一个人整天在外面跑&#xff0c;感染的几率最大。所以最后想了一下&#xff0c;决定先在外面租个房子&…

零基础学编程,怎么开始学习?

编程零基础的话&#xff0c;我先建议你看一些经典的书籍&#xff0c;抑或是通俗易懂的计算机常识书。 这几本书各有千秋&#xff0c;我参考了我自己尝试过的几种方法&#xff0c;可以为你选择最适合你的学习方法提供一种参考。首先要判断你的决心有多大&#xff0c;一则花费金…

华为云会议,开会就是如此简单

现在工作节奏加快&#xff0c;高效沟通&#xff0c;快速决策&#xff0c;立刻执行成为组织提升整体效益的关键&#xff0c;而会议作为企业之间重要的沟通工具&#xff0c;被广泛的应用于日常工作中&#xff0c;云会议更可以跨越时空的限制&#xff0c;更为方便快捷。 华为云会议…

UML类图语法介绍

UML类图语法介绍一 官方定义基本介绍UML 图分类建模工具二 六大关系2.1 依赖关系代码体现UML图示2.2 泛化关系代码体现UML图示2.3 实现关系代码体现UML图示2.4 关联关系代码体现UML图示2.5 聚合关系代码体现UML图示2.6 组合关系代码体现UML图示一 官方定义 UML - Unified model…

设置开机自动启动jupyter notebook及远程访问的方式

jupyter notebook远程访问设置 1.安装 conda activate abc #激活虚拟环境[不在虚拟环境中可以省略此步骤] pip install jupyter #安装 jupyter notebook --generate-config #生成配置文件备注&#xff1a;配置文件的所在路径一般为 ~/.jupyter/jupyter_notebook_config.py 注…

四、ArrayList底层源码详解

文章目录特点底层源码分析创建无参构造器有参构造器传入集合的有参构造器扩容注意new ArrayList(0)菜鸟教程ArrayList讲解特点 可以加入控制null(可加入多个)底层是数组实现的ArrayList基本等同于Vector&#xff0c;除了ArrayList是线程不安全(执行效率高)看源码(没有synchron…

“消费盲返”就像一盒巧克力,你永远不知道下一颗是什么味道

如今电商平台竞争越来越激烈&#xff0c;传统电商直播短视频带货拼团带货内卷严重&#xff0c;我就经常刷抖音某多多看见各种优惠&#xff0c;感觉以前在实体店上面买东西都是冤大头&#xff0c;但是随之而来的是商家引流成本越来越高。 这些年来&#xff0c;互联网迅猛发展&am…

Allegro如何批量添加丝印文字操作指导

Allegro如何批量添加丝印文字操作指导 当需要在PCB上需要大量添加丝印文字的时候,是个较大的工作量,Allegro支持批量添加丝印文字到PCB上,如下图 具体操作如下 新建一个文本文档在文本文档中添加好需要的文字

红黑树详解

1.概念 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或Black。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制&#xff0c;红黑树确保没有一条路径会比其他路径长出俩倍&#xff0c;因而是接近…

漏洞丨cve2010-3333

作者&#xff1a;黑蛋 一、漏洞简介 cve-2010-3333是一个Office 2003 的栈溢出漏洞&#xff0c;其原因是在文档中读取一个属性值的时候&#xff0c;没有对其长度验证&#xff0c;导致了一个溢出&#xff0c;看着很简单的一个漏洞&#xff0c;却又有点恶心人。 二、漏洞环境 …

Python学习基础笔记五十五——接口类和抽象类

# 设计模式&#xff1a; 《设计模式》 接口类&#xff1a;Python原生不支持&#xff1b; 抽象类&#xff1a;Python原生支持的。 例&#xff1a; from abc import abstractmethod, ABCMetaclass Payment(metaclassABCMeta): # 元类 默认的元类是 type 规范类abstract…

MyBatis学习 | 全局配置文件

文章目录一、简介二、各个标签2.1 properties&#xff08;属性&#xff09;2.2 settings&#xff08;设置&#xff09;2.3 typeAliases&#xff08;类型命名&#xff09;2.4 typeHandlers&#xff08;类型处理器&#xff09;2.5 plugins&#xff08;插件&#xff09;2.6 enviro…

电容这段走线影响这么大?

公众号&#xff1a;高速先生 作者&#xff1a;李远恒 一日&#xff0c;来了一个电源仿真项目&#xff0c;雷工像往常一样熟练的打开了PCB文件&#xff0c;先是例行查板。不查不要紧&#xff0c;一查还真有问题&#xff0c;话不多说直接上图&#xff1a; 定睛一看&#xff0c;这…

星火计划学习笔记——参考线平滑算法解析及实现(以U型弯道场景仿真调试为例)

文章目录1. Apollo参考线介绍1.1 参考线的作用1.2 导航规划的路线1.3 为什么需要重新生成参考线1.4 ReferenceLine数据结构1.5 ReferencePoint数据结构1.6 参考线处理流程1.7 参考线生成2. 参考线平滑算法2.1 参考线平滑算法总览2.2 参考线平滑算法流程2.2.1 设置中间点anchor …

康希通信冲刺科创板上市:上半年收入2亿元,计划募资约8亿元

12月21日&#xff0c;格兰康希通信科技&#xff08;上海&#xff09;股份有限公司&#xff08;下称“康希通信”&#xff09;在上海证券交易所递交招股书&#xff0c;准备在科创板上市。本次冲刺科创板上市&#xff0c;康希通信计划募资7.82亿元。 据贝多财经了解&#xff0c;康…

Word处理控件Aspose.Words功能演示:在 Java 中将 DOC 或 DOCX 转换为 PNG

aspose.words是一种高级Word文档处理API&#xff0c;用于执行各种文档管理和操作任务。API支持生成&#xff0c;修改&#xff0c;转换&#xff0c;呈现和打印文档&#xff0c;而无需在跨平台应用程序中直接使用Microsoft Word。此外&#xff0c; Aspose API支持流行文件格式处…

【C++入门基础(上)】

Cross the stars over the moon to meet your better-self. 目录 1 命名空间 1.1 命名空间定义 1.2 命名空间使用 1.2.1 加命名空间名称及作用域限定符 1.2.2 使用using将命名空间中成员引入 1.2.3 使用using namespace 命名空间名称引入 2 C输入&&输出 3 缺省参数…

Mentor-dft 学习笔记 day47-On-Chip Clock Controller Design Description

On-Chip Clock Controller Design Description有三种类型的片上控制器设计&#xff1a;standard, parent, and child。可以根据设计要求选择它们。使用OCC时&#xff0c;必须考虑本节中讨论的设计元素。The Standard OCC 标准OCC为快速捕获提供快速时钟&#xff0c;为换档和慢速…

Java字节流的使用:字节输入/输出流、文件输入/输出流、字节数组输入/输出流

InputStream 是 Java 所有字节输入流类的父类&#xff0c;OutputStream 是 Java 所有字节输出流类的父类&#xff0c;它们都是一个抽象类&#xff0c;因此继承它们的子类要重新定义父类中的抽象方法。 这里首先介绍上述两个父类提供的常用方法&#xff0c;然后介绍如何使用它们…

springboot整合shiro + jwt + redis实现权限认证(上手即用)

目录前言项目结构依赖导入建数据库表建表语句使用插件生成增删改查添加MyRealm添加ShiroConfig添加JwtFilterJWT相关得类JwtTokenJwtAudienceJwtHelper添加BeanFactory只贴出主要得类&#xff0c;具体得可以看我的gitee&#xff0c;接口都自测过的。前言 最近项目中涉及到使用…