函数递归调用

一、函数参数与函数的值

1.在定义函数中指定的形参,在未出现函数调用时,它们并不占用内存中的存储单元,只有在发生函数调用时,函数中的形参才会被分配内存单元。在调用结束后,形参所占的内存单元也会被释放。
2.实参可以是常量、变量或表达式。在被定义的函数中,必须指定形参的类型,实参与形参的类型应当相同或赋值兼容。
3.实参向形参的数据传递是“值传递”,单向传递,只能有实参传给形参。函数的形参相当于实参的复制,对形参进行修改不会对实参产生影响。解决办法为传地址。
4.无法在被调函数中修改主调函数的参数。
5.函数传参默认自右向左。
6.栈区:栈区由编译器自动分配释放,由操作系统自动管理,无须手动管理。栈区上的内容只在函数范围内存在,当函数运行结束,这些内容也会 自动被销毁。栈区是先进后出原则。
堆区:堆区由程序员分配内存和释放。

字符串常量区:字符串、数字等常量存放在常量区。Const修饰的全局变量存放在常量区。程序运行期间,常量区的内容不可以被修改。
静态区(全局区):通常是用于那些在编译期间就能确定存储大小的变量的存储区,但它用于的是在整个程序运行期间都可见的全局变量和静态变量。全局区有.bss段和.data段组成,可读可写。
        bss段:未初始化的全局变量和未初始化的静态变量存放在**.bss段**。.bss段不占用可执行文件空间,其内容由操作系统初始化。
        data段:已初始化的全局变量和静态变量存放在.data段。data段占用可执行文件空间,其内容有程序初始化。
代码区:程序执行代码存放在代码区,其值不能修改(若修改则会出现错误)。字符串常量和define定义的常量也有可能存放在代码区。

二、函数嵌套调用

本质:函数的跳转

                            6053c1640b144823b06c1f7033d2a05f.png

判断每个月有多少天数

int isLeapyear(int year)
{
	return year % 4 == 0 && year % 100 != 0 || year % 400 == 0;
}
int daysOfTheMonth(int year, int month)
{
	int days[] = {31, 0, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
	
	assert(month >= 1 && month <= 12);
/*	if(!(month >= 1 && month <= 12))
	{

		return -1;
	}*/

	if(month != 2)
	{
		return days[month - 1];
	}
	return isLeapyear(year) ? 29 : 28;
}

三、函数递归调用

        在调用一个函数的过程中又直接或间接的调用该函数,称为函数的递归调用。函数调用分为直接递归和间接递归

用递归函数输出斐波那契数列

int fn(int n)
{
	if(1 == n)
	{
		return 1;
	}
	else
	{
		return fn(n - 1) * n;
	}
}

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

int main(void)
{
	printf("%d\n", fib(40));
	return 0;
}


注意:使用函数时要降低函数的耦合性,提高函数的复用性。

 

  • 11
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值