函数嵌套调用,链式访问,声明与定义,递归等等

Tips

1.函数是一个大型程序中的某部分代码,有一个或多个语句块组成,负责完成某项特定任务,相当于“外包”。具备相对的独立性。函数提供对过程的封装与细节的隐藏。

2.数组可以不指定元素的个数,他它会根据初始化的内容来确定。

3.传值调用实参与形参占有不同的内存空间,对形参的修改不影响实参。当传址调用时,函数和函数外面的变量建立了联系。

4.计算数组元素个数的方法 int sz =sizeof(arr)/sizeof(arr[0]),先计算整个数组的内存大小,单位是字节;再计算数组第一个元素的内存大小,单位是字节。

5.函数参数里面可以传数组,也可以传入多个参数。

6.strlen()求字符串长度,也可以求字符数组长度.

7.在函数里面尽量少用全局变量,还不如自己定义一个局部变量,把全局变量的东西传到局部变量里面来操作。这样可以使全局变量不受影响。

8.当函数把实参传到形参里面去时,就相当于等号这一步操作,这个理解很方便。

9.函数的返回类型不写的时候,默认返回的是int类型。

10.输入无符号的整数用%u。

函数的嵌套调用

1.嵌套调用指的是我这个函数在使用的过程中里面会调用你这个函数。我调用,你调用他,大家互相调用。

 

2.如果仔细去观察一下代码,你会发现任何代码都是由函数组成的。函数与函数之间可以根据实际的需求进行组合,也就是互相调用。

3.函数可以嵌套调用,但是不能嵌套定义。也就是说在一个函数体内不能去定义另外一个函数。每个函数都是平等的,定义函数的时候得放到函数外头去。

函数的链式访问

1.函数的链式访问:可以把一个函数的返回值作为另外一个函数的参数。

 

 

再次强调一下,printf()的返回值与功能是不一样的,printf()函数返回的是打印在屏幕上字符的个数,它的功能是打印相应的内容,返回值与功能是不一样的。


 

函数的声明与定义

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

int add (int x,int y);

 函数的定义(函数的定义是指函数的具体实现,交代函数的功能实现)

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

1.代码在进行编译的时候是要扫描。扫描的时候是从上往下扫描,一旦编译器碰到没有出现过的“名称”的时候,就会先爆出一个警告,尽管你是在之后才去定义的。

2.一个函数在使用之前须对其进行声明或者定义。否则编译器扫描到这边的时候,他是不认得的。一种写法是在前面进行声明,然后在后面进行定义。还有一种写法是干脆直接在前面定义(函数定义也是一种特殊的声明)。

3.但事实上函数的声明与定义都不是这样用的。上面所说的只是他的一种应用场景或者语法展示。而真正在工程里面,首先先去新建一个头文件,然后再去新建一个源文件。将函数的定义放在源文件,将函数的声明放在头文件。然后如果想在工程里面使用函数呢,你只要做一件事情就可以了,#include "刚才新建的头文件"。

4.因此以后关于函数的声明一般都放在头文件中,函数的定义或者实现都是放在源文件里面。

比如说计算两个数的最大公因数

 

 

刚才新建的那个源文件与新建的头文件合起来成为一个模块

5.这边还要注意一下,两个.c源文件之间的函数要互相用,可以用extern声明外部符号来实现。
那为什么要像这样拆成三个文件呢?首先这样就可以模块化开发(分工),也可以做到代码的隐藏。

 

6.函数是这样,变量的声明与定义其实与函数是一样的。

7.全局变量不初始化的时候,默认是0。

函数递归

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

2.函数内部自己调用自己就可以理解为递归。

但当无穷递归下去的时候,最后程序就会崩掉。原因是每一次调用函数,都会为本次函数在内存的栈区上开辟一块内存空间。当函数无穷递归下去时(也就是说在内部无穷的自己调用自己),总有一天会把内存的栈区耗干,这时候就会栈溢出(递归层次过深,将栈空间耗干,就会出现栈溢出的情况),这就会导致程序崩溃。因此,无穷递归一定要避免。

3.递归事实上就是一个过程或函数,在其定义或说明中有直接或者间接调用自身的一种方法。利用递归大大减少了程序的代码量,其实他就是通常把一个大型复杂的问题层层转化为一个语言问题相似的,规模较小的问题来求解可以想象数列递推公式中的第n项与第(n-1)项。
4.别看用递归只有几行代码。但其实里面执行了大量重复的操作。递就是递推的意思,归就是回归的意思。要理解这个,也可以用数学数列里面的递推公式去理解。

5.递归的两个必要条件,1.首先递归存在限制条件,当满足这个限制条件的时候,递归不再继续。     2.每次递归调用之后会越来越接近这个限制条件。

6.要理解函数递归,其实只要抓住这么几个点:你就先认为那个函数是成立的(虽然还没写完),然后把An过程拆为A(n-1)过程与其他过程,用数列的递推公式来理解最为直观形象。

接下来还是不得不普及一个极其重要的知识

1.数组名其实就是该数组第一个元素的地址(地址与指针是同一个概念),因此一个函数如果参数里面传入的是一个数组,那么在函数定义的时候,接收参数可以用指针(但别忘了,这个指针指的是有且仅有数组的第一个元素

2.比如说arr是一个数组名,其实也就是数组首元素的地址,如果想让这个指针往后走一步,该怎么操作?就“arr ++”或者"arr+=1"

3.

字符指针+1,其实就是它向后跳一个字符,就是向后跳一个字节。
整形指针+1,其实就是它想后跳一个整形,也就是说向后跳四个字节(因为一个整形占四个内存单元,每个内存单元是一个字节)


 

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

絕知此事要躬行

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值