C语言之初识函数

这篇具有很好参考价值的文章主要介绍了C语言之初识函数。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1.函数是什么

在维基百科中函数的定义:子程序

在计算机科学中,子程序是一个大型程序中的部分代码,是一个或多个语句块组成。它负责某块特定任务,相对于其它代码而言,具有相对的独立性。
一般会有输入参数并有返回值,提供对过程的分装和隐藏,这些代码通常被集成为软件库。

2.库函数

库函数,那库函数为什么要存在呢?为什么要有库函数呢?
1.就比如说我们在编写C程序是要频繁的输出某一个结果,想要把这个结果打印到我们的电脑屏幕上,这个时候我们就需要用到printf函数,即将信息按照某一个是打印到我们的屏幕上。
2.再比如说我们要求一个数的开平方根,这时候我们就需要用到sqrt函数。
3.在编程时我们有时也需要计算n的k次方,这是我们就可以用pow函数。
类似于上面我们要实现某些特定的功能,它们并不是业务性的代码,我们在开发的过程中每个程序员都有可能用的到,甚至是频繁的使用,为了提高可移植性和提高程序的效率,所以C语言的基础库中提供了一系列类似上述的库函数,方便程序员进行软件开发。
所以,我们如果想实现一些功能时,直接调用库函数就可以了,人家库函数已经为我们准备好了,就不需要我们自已编写程序来实现特定的功能了。
下面举一些C语言常用的库函数:

IO函数    input  output函数
字符串操作函数  strlen   strcmp
内存操作函数   memcpy  memmove  memset
时间/日期函数    time
数字函数  sqrt  pow
其他库函数

下面是学习库函数的一些网站:
www.cplusplus.com
https://en.cppreference.com (英文版)
https://zh.cppreference.com (中文版)

3.自定义函数

刚刚提到了库函数,但是库函数中的功能毕竟是有限的,是有局限的,它不可能解决我们所有需要的功能,所以说如果库函数能解决所有问题,那我们程序员是来干什么的呢?这个时候就需要自定义函数了,即自已编写一个函数使其能够实现特定的功能。自定义函数和函数名一样,有函数名、返回值类型、函数参数
函数的组成:

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

下面我们要写一个函数要求两个数中的最大值,请看:

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

C语言之初识函数
当然我们也可以把上述代码简化一些:

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

再来举一个简单的例子:

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

上述代码中的void的什么意思呢?我们还是拿代码来举例:

C语言之初识函数
C语言之初识函数

现在,我们想写一个swap函数来实现两个数的交换,我们这么写可不可以呢?请看:

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

如果你认为上述的代码可是实现a和b的交换,即交换之后的结果应该是a变为20,而b变为10,但结果真是如此吗?请看运行结果:
C语言之初识函数
我们会惊奇的发现,忙活了半天但是我们写的swap函数好像并没有起到交换的作用,即我们写出来一个bug,只能说明我们写的代码有问题。
我们先通过调试看看能不能找这其中的问题:
C语言之初识函数
我们可以发现a,b的地址和x,y的地址不相同。即a,b,x,y都拥有属于自己的一块内存空间,我们只是把a和b的值通过swap函数传给了x,y后,把x,y中的值进行了交换,既然是把x,y的值进行了交换,那么就对a,b没有产生任何影响。
在这里我们把a和b叫做实参,即实际参数,而x和y叫做形式参数,但我们把实参a和b的值传给形参后,不要忘了形式参数x和y也拥有属于自己的一块空间所以改变形参(在这里指的是交换x和y的值)不会对实参产生任何的影响因此当函数调用的时候,实参传给形参,这时形参是实参的一份临时拷贝,对形参的修改不影响实参。
既然上述代码不能实现两数交换的话,那我们不妨还一种思路(通过地址的方式(指针)将两者之间建立联系),在这之前先看一段代码:
C语言之初识函数
所以,我们可以通过地址的方式将两者之间建立联系。请看代码:
C语言之初识函数

***相当于我们可以通过地址的方式,从swap函数中远程的操作主函数中a和b的值。***但是要注意a和b的地址肯定是没有变的,只不过是把a和b的地址所指向空间的内容改变了
所以,那以后什么情况下我们需要把地址传过去呢?当这个函数内部可能会改变主函数中某些信息的时候,此时我们就可以把地址传过去,记住这种思路。

4.函数参数

4.1实际参数(实参)

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

4.2形式参数(形参)

形式参数是指函数名后括号中的变量,因为形式参数只有在函数被调用的过程中才实例化(分配内存单元),所以叫形式参数。
形式参数当函数调用完成之后就被销毁了。因此形式参数只有在函数中有效。
还是以下面代码为例:

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

那么请问形式参数x和y会不会占用内存空间呢?
只有当我们调用get_max函数时,我们才会给形式参数x和y分配空间,在调用get_max函数之前,形式参数x和y是不会占用空间的,它只是形式上存在而已。
另外补充一点:形参实例化的意思就是当我们把实参的值传递给形参并且创建形参的这个过程(也可以理解为为形式参数创建空间的过程)叫做形参实例化实参实例化之后其实相当于实参的一份临时拷贝

5.函数调用

5.1传值调用

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

5.2传址调用

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

5.3练习

写一个函数求100到200之间的素数:

#include<stdio.h>
#include<math.h>
int isprime(int n)
{
	for (int j = 2; j <= sqrt(n); j++)
	{
		if (n % j==0)
			return 0;
	}
	return 1;
}
int main()
{
	for (int i = 100; i <= 200; i++)
	{
		//判断是否为素数,是则返回1,否则返回0
		if (isprime(i))
		{
			printf("%d ", i);
		}
	}
	return 0;
}

写一个函数判断一年是不是闰年:

#include<stdio.h>
int is_leap_year(int n)
{
	//是闰年返回1,不是闰年返回0
	if (n % 4 == 0 && n % 100 != 0 || n % 400 == 0)
		return 1;
	else
		return 0;
}
int main()
{
	for (int i = 1000; i <= 2000; i++)
	{
		if (is_leap_year(i))
			printf("%d ", i);
	}
	return 0;
}

写一个函数,实现一个整型有序数组的二分查找(也称为折半查找):

#include<stdio.h>
int binary_search(int arr[], int k, int sz)
{
	int left = 0;
	int right = sz - 1;
	while (left <= right)
	{
		int mid = (left + right) / 2;
		if (arr[mid] < k)
		{
			left = mid + 1;
		}
		else if (arr[mid] > k)
		{
			right = mid - 1;
		}
		else
			return mid;
	}
	return -1;//找不到了
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	int k = 8;
	//如果能找到就返回下标,找不到则返回-1
	int ret = binary_search(arr, k, sz);
	if (ret == -1)
		printf("找到了\n");
	else
		printf("找到了,下标为%d", ret);
	return 0;
}

写一个函数,每调用一次这个函数,num的值就加1:
方式一:

#include<stdio.h>
void Add(int* p)
{
	*p = *p + 1;
}
int main()
{
	int num = 0;
	Add(&num);
	printf("%d\n", num);
	Add(&num);
	printf("%d\n", num);
	Add(&num);
	printf("%d\n", num);
	return 0;
}

方式二:

#include<stdio.h>
int Add(int n)
{
	return n + 1;
}
int main()
{
	int num = 0;
	num = Add(num);
	printf("%d\n", num);//1
	
	num = Add(num);
	printf("%d\n", num);//2

	num = Add(num);
	printf("%d\n", num);//3

	return 0;
}

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

6.1嵌套调用

程序都是由函数组成的。
函数和函数之间可以根据实际的需求进行相互的组合,也就是嵌套调用。
举一个简单例子:

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

C语言之初识函数
再来举个小例子:

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

7.函数的声明和定义

7.1函数声明

1.告诉编译器有一个函数叫什么,参数是什么,返回类型是什么,但是具体该函数存不存在,不是由函数声明决定的。
2.函数的声明一般出现在函数的使用之前,要满足先声明后使用
3.函数的声明一般要放在头文件中。
在有些书中,是这样写的:
C语言之初识函数
在这里再次强调上述代码是不规范,不正确的。尽管程序能够运行起来。当上述程序运行起来时,编译器会爆出警告:即Add未定义,那为什么会出现这样的警告呢?是这样的:***因为代码在进行编译时,要进行代码的扫描,而代码进行扫描时是从前往后进行扫描的。***所以说,当我们先用函数而后进行函数声明时,编译器就会进行警告。

故,我们应该这样来写代码(即先进行函数声明和定义,而后使用):

#include<stdio.h>
//函数的声明
int Add(int x, int y);//注意不要丢掉分号
int main()
{
	int a = 10;
	int b = 20;
	int ret = Add(a, b);
	printf("ret=%d\n", ret);
	return 0;
}
//函数的定义
//函数的声明是一种特殊的函数定义
int Add(int x, int y)
{
	return x + y;
}

也可以这样:

#include<stdio.h>
int Add(int x, int y)
{
	return x + y;
}

int main()
{
	int a = 10;
	int b = 20;
	int ret = Add(a, b);
	printf("ret=%d\n", ret);
	return 0;
}

但是实际上呢,函数的声明和定义不是这样来使用的,上述代码和举例只是我们的一个语法展示,那么真正在工程中函数的声明和定义应该是怎样用的呢?比如:
C语言之初识函数
C语言之初识函数
C语言之初识函数
这与变量的声明和定义有着相似之处,比如:
C语言之初识函数

所以,一般函数的声明放到.h头文件中,而具体函数的实现(也可以说函数的定义)放到.c的头文件中去。这才是函数的声明和定义应该有的一种形式。
***那为什么要这么麻烦把.h.c文件分开写呢?,其一是方便模块化开发,其二是代码的隐藏。


总结:函数的声明放到.h头文件中,函数的定义(或函数的实现)单独放到.c源文件中。在未来遇到稍微复杂的代码我们就可以分文件去写。

8.函数递归

8.1什么是递归?

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

举一个很容易理解的递归:
C语言之初识函数
只不过是这个递归陷入死循环了,所以说这是一个错误的递归,最终这个程序会因为栈溢出导致崩溃而停止运行。

8.2递归的两个必要条件

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

8.2.1练习

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

#include<stdio.h>
void Print(n)
{
	if (n > 9)
	{
		Print(n / 10);
	}
	printf("%d ", n % 10);
}
	
int main()
{
	unsigned int num = 0;
	scanf("%d", &num);
	//写一个函数把num的每一位打印出来
	Print(num);
	return 0;
}

我们发现每次递归都会接近n>9这个限制条件,所以递归最终会停下来。
下面再来看一个练习:编写函数不允许创建临时变量,求字符串长度

#include<stdio.h>
int my_strlen(char* str)
{
	int count = 0;
	while (*str != '\0')
	{
		count++;
		str++;//字符指针+1,先后跳一个字符
	}
	return count;
}
int main()
{
	char arr[] = "helloworld";
	int len = my_strlen(arr);
	printf("len=%d", len);
	return 0;
}

但是题目要求是不创建临时变量,而上述代码创建了临时变量count,这个时候就得找其他方法。请看:

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

我们也会发现每次调用my_strlen函数时都会接近递归条件*str!='\0',因此我们要熟练掌握这种思想,在今后会为我们学习提供另一种思路。
另外记住一句话:递归=递推+回归

8.3递归与迭代

写一个函数求n的阶乘:
方式一(递归思想):

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

方式二(迭代思想):

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

写一个函数求第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);
	int ret = fib(n);
	printf("%d\n", ret);
	return 0;
}

在运行上述代码时,我们肯定会发现一个问题:
1.在使用fib这个函数计算第五十个斐波那契数时会特别的消耗时间。
2.在使用factorial函数求10000的阶乘时(不考虑结果的正确性),程序会崩溃。
那如何解决上述问题?

1.将递归改为非递归。
2.使用static对象代替nonstatic局部对象,在递归函数设计中,可使用static对象替代nonstatic局部对象(即栈对象),这不仅仅可以减少每次递归调用和返回时产生和释放nonstatic对象的开销,而且static对象还可以保存递归调用的中间状态,并且可用各个调用层所访问。

虽然递归可能会存在栈溢出的情况,运算速度比较慢,但递归的一个非常好的优势是它的代码少,同样一个题目如果用递归的方式可能几行就写完了,而如果用非递归的方式可能就需要大量的代码,用起来也比较复杂。

方式二(非递归):

#include<stdio.h>
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("ret=%d", ret);
	return 0;
}

最后提示:
1.许多问题是以递归的形式解释的,这只是因为它比非递归的形式更为清晰。
2.但是这些问题的迭代实现往往比递归实现效率更高,虽然代码的可读性稍微差一些。
3.当一个问题相对复杂,或难以用迭代实现时,此时递归实现的简洁性便可以弥补它所带来的运行时开销。

函数递归经典题目(这里不展开说明):
1.汉诺塔问题
2.青蛙跳台阶问题
最后,本文直到知道这里就结束了,望对大家的学习有所帮助。再次感谢!!!文章来源地址https://www.toymoban.com/news/detail-463114.html

到了这里,关于C语言之初识函数的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 初识C语言·内存函数

    目录 1 memcpy的使用和模拟实现 2 memmove的使用和模拟实现 3 memset的使用和模拟实现 4 memcmp的使用和模拟实现 紧接字符串函数,出场的是第一个内存函数memcpy。前面讲的字符串函数是专门干关于字符串的事的,而这个函数可以干strcpy一样的事,但是区别就是它碰到\\0也会继续复制

    2024年01月20日
    浏览(57)
  • 【C语言】初识C语言之函数不安全

    C语言程序的基本框架 程序显示一闪而过问题的解决 函数不安全问题的解决 一、第一个C语言程序 解读: 1、main函数:函数的书写格式“函数名()”,C语言是从主函数的第一行开始执行,主函数是程序的入口,mian函数,必须有且仅有一个。 2、如何证明main函数时程序的入口:

    2024年02月08日
    浏览(31)
  • 数据结构和算法学习记录——初识二叉树(定义、五种基本形态、几种特殊的二叉树、二叉树的重要性质、初识基本操作函数)

    目录 二叉树的定义 二叉树具体的五种基本形态 1.空树 2.只有一个节点 3.有左子树,但右子树为空 4.有右子树,但左子树为空  5.左右两子树都不为空 特殊二叉树 斜二叉树 满二叉树  完全二叉树 二叉树的几个重要性质 初识二叉树的几个操作函数  二叉树T: 一个有穷的节点

    2024年02月03日
    浏览(61)
  • 【C++学习】类和对象 | 拷贝构造 | 探索拷贝构造函数为什么需要引用传参 | 深拷贝 | 初识运算符重载

    上一篇文章我们开始学习类内的默认成员函数, 这里是传送门,有兴趣可以去看看:http://t.csdn.cn/iXdpH 这篇文章我们继续来学习类和对象的知识。 目录 写在前面: 1. 拷贝构造 2. 拷贝构造函数为什么需要引用传参? 3. 深拷贝 4. 初识运算符重载 写在最后: 我们在创建一个对

    2024年02月11日
    浏览(53)
  • CRM百科 | CRM是什么?

    究竟什么是 CRM ?企业为什么要用 CRM ? CRM 到底有什么作用? CRM 是Customer Relationship Management(客户关系管理)的缩写,是一种通过对客户进行跟踪、分析和管理的方法,以增加企业与客户之间的互动和联系,提高企业与客户之间的互信,从而实现企业的业务目标。 CRM是一种

    2024年02月14日
    浏览(37)
  • 创建个人百度百科需要什么条件?

    互联网时代,创建百度百科词条可以给个人带来更多的曝光和展现,相当于一个镀金的网络名片,人人都想上百度百科,但并不是人人都能创建上去的。 个人百度百科词条的创建需要满足一定的条件,今天 伯乐网络传媒 就来给大家聊聊这个话题,看你到底能不能上百度百科

    2024年04月16日
    浏览(44)
  • 什么是数据库中的函数(库函数与自定义函数)

    目录 数据库的库函数 一、聚合函数: 1、count: 二、日期时间函数: 1:now: 2:data 3:time 4:date_format 三、常用的库函数 1:upper 2:lower 3:substring 4:round 5:length 6:concat 7:database 8:user 9:power 四、其他函数: 自定义函数的创建: 自定义函数function 1、 函数创建 2、函数调用 3、自定义

    2024年02月02日
    浏览(68)
  • 什么原因导致百度百科建立一直审核不通过?

    百科词条对网络营销实在是太重要了,不管是个人还是企业想在网上开展业务,都必要建立百科词条。自己动手编辑百科词条,搞个几十次也审核不过的情况比比皆是。 为什么百度百科总是审核不通过?百度官方发表过声明表示百度百科词条是人人都可以编辑的,并且都是免

    2024年02月19日
    浏览(45)
  • IT知识百科:什么是分布式云?

    在当今信息技术高速发展的时代,云计算已经成为了企业和个人的重要组成部分。而在云计算领域中,分布式云是一种重要的架构模式,它允许资源的分散部署和管理,以实现高可用性、可伸缩性和弹性的服务提供。 本文将详细介绍什么是分布式云,它的特点、优势以及在实

    2024年02月06日
    浏览(46)
  • 【C语言】函数的定义及调用

            刚刚结束了数组的介绍,有需要的猿友可以去看我之前的文章,从这篇文章开始,进行函数相关的介绍,代码均来自VS编译环境下。 目录 一、定义函数的方法  二、调用函数 1.函数调用的形式

    2024年02月05日
    浏览(43)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包