在以前的c语言学习中我们解题经常会使用到函数,那么本节将介绍在Java里面的函数(方法),这一节博客呢主要讲了在Java中方法定义,使用,传参和重载,以及递归的学习,帮助你能有更好的解题思路和方法!
1.1 什么是方法(method)
方法就是一个代码片段. 类似于 C 语言中的 “函数”。
1.2 方法定义
语法格式:
举例方法的使用:
1.两个数的相加:
上面的就是方法,和c语言中的函数一样,int就是方法返回值类型,add就是方法名,(int a,int b)就是形式参数列表。那么我们想要调用这个方法,就必须做到一下两点:
2.闰年的判断:
注意:此处程序运行的时候,遇到return就直接结束了,所以不需要else。
注意事项:
1.3 方法调用的执行过程
方法调用过程:
调用方法—>传递参数—>找到方法地址—>执行被调方法的方法体—>被调方法结束返回—>回到主调方法继续往下执行
当方法被调用时(方法在Java中本质上都是按值传递),会在栈上开辟一块内存,实参传递到形参上面之后,遇到return返回函数的时候,方法开辟的内存就会被销毁。
注意:
我们这里再举一个之前提到的例子:计算1!+2!+3!+4!+5!
但是这次我们调用方法来写
我可以详细讲述一下步骤,fac方法就是计算n的阶乘,facSum方法是在fac方法外嵌套一个循环,让它能计算一个任意的数值,然后主函数调用的时候就可以计算了
1.4 实参和形参的关系(重要)
方法的形参相当于数学函数中的自变量,Java中方法的形参就相当于sum函数中的自变量n,用来接收sum函数在调用时传递的值的。形参的名字可以随意取,对方法都没有任何影响,形参只是方法在定义时需要借助的一个变量,用来保存方法在调用时传递过来的值。
代码示例: 交换两个整型变量
输出结果:
我们可以清晰的看到,他们在进行一系列操作之后并没有交换成功,那么问题出在哪呢?我们可以回忆起以前学的c,函数调用可以取地址,但是在Java里并没有取地址,Java做不到,在栈上的地址是拿不到的,由于此处我们学的东西还较少,所以目前无法做得出来,后面更新数组我们就可以做这一题了,到时候会再讲一次这一题。
在Java中,实参的值永远都是拷贝到形参中,形参和实参本质是两个实体,在swap方法调用时,只是将实参a和b中的值拷贝了一份传递给了形参x和y,因此对形参x和y操作不会对实参a和b产生任何影响。对于基础类型来说, 形参相当于实参的拷贝. 即 传值调用
1.5 没有返回值的方法
方法的返回值是可选的. 有些时候可以没有的,没有时返回值类型必须写成void。
一般情况下我们建议要返回值,因为没有返回值类型的话,我们在计算或者引用了方法之后,想要再去对方法返回的值进行操作的时候,它是不支持的。
2.1 为什么需要方法重载
经过上面的学习我们可以知道,返回值类型不同是无法调用方法的,比如我们方法里的返回值是int类型,然而我们实参想传递一个double类型的数,这时候就会产生错误,无法编译。那么我们这时候就要学习方法的重载了,它可以有效帮我们解决这个问题。
2.2 方法重载概念
在Java中,如果多个方法的名字相同,参数列表不同,则称该几种方法被重载了。
如下列代码:
从上面我们可以看到并没有报错,而且可以推断出方法重载的条件:
2.3 方法签名
方法签名即:经过编译器编译修改过之后方法最终的名字。
具体方式:方法全路径名+参数列表+返回值类型,构成方法完整的名字。
一段重载代码经过编译之后,然后使用JDK自带的javap反汇编工具查看,具体操作:
具体操作:
一段重载代码生成之后,在文件夹里找到相应的IDEA目录文件名
就好比我来说,我生成了日期目录
然后打开目录,找到out文件夹
一直顺势点进去之后,会发现有你再src命名的文件夹,后缀名为XXX.class
按住Shift+鼠标右键
点进去,我们会看到以下画面
然后输入javap -v 字节码文件名就可以看到如下反汇编代码了
I 代表 int,D 代表 double,那么我们也可以详细做一做了解关于方法签名中的特殊符号。
3.1贴近生活中的事情
生活中的例子如下图,一个女孩子拿着镜子对着镜子照,于是形成了镜子中间有镜子,一直循环。
这里的重要信息就是:自身中又包含了自己,该种思想在数学和编程中非常有用,因为有些时候,我们遇到的问题直接并不好解决,但是发现将原问题拆分成其子问题之后,子问题与原问题有相同的解法,等子问题解决之后,原问题就迎刃而解了。
3.2 递归的概念
一个方法在执行过程中调用自身, 就称为 “递归”.
例如:我们写一个这样的代码
无限的套用自己,也算是递归吗?我们运行起来,发现出了错误
我们可以看到出现了栈溢出错误,因为这个循环没有终止条件,导致它在栈帧上循环开辟空间,最后栈帧内存装不下了,就会导致溢出。
总结上面的例子,我们引出了如下递归条件:
3.3 递归练习
例一:按顺序打印一个数字的每一位(例如 1234 打印出 1 2 3 4)
这一题的思路需要很明确,如下图示范:
由图可知我们每次想得到一位数字,都要先模10再除10,所以循环公式找到了
但此题需要注意的一点就是图中红色圆圈这一点,由我上面提到的,一般序列打印的话,是反向打印,如果按照正常来,输入123,结果会是321,与预期不符,所以红色圈这里两句顺序很重要,具体的草图我就不做图解了,有不理解的小伙伴都可以来问我哦~欢迎大家一起讨论
例二:递归求 1 + 2 + 3 + … + 10
思路:sum(10) = 10 + sum(9) --> sum(9) = 9 + sum(8)…循环公式就找到了
例三: 写一个递归方法,输入一个非负整数,返回组成它的数字之和. 例如,输入 1729, 则应该返回1+7+2+9,它的和是19
思路:
由图可知每次都需要除上一个10,再模去一个10,就能得到相应的数字,循环公式就找到了。
例四:求斐波那契数列的第 N 项(0、1、1、2、3、5、8、13、21、34、……)
我们可以发现斐波那契数列从第二项开始,每一项都是前两项之和,所以通式找到了
当等于前两项的时候,它是需要我们去打印的,从第三项开始之后电脑可以算出来
但是第n项比较大时,会发现程序越来越慢,经过分析,它在执行的时候,做了很多重复的计算,在计算第n项时,其中第n-1项算一次,第n-2项又算了一次,所以当n越来越大时,程序会进行很慢,可以知晓这也是递归的一个弊端。
递归特点:
所以此处我们又引出来了循环
就拿例四来说:
我们前面两个if语句就是给第一项和第二项定值,后面的for循环就是让每次循环的时候,第三个数都能成为第一项和第二项的和,避免了上面递归的重复计算,所以这一题循环在时间,空间和效率上更胜一筹。
循环的特点:
那我们方法的使用就到此为止啦,还是比较简单的,大家有什么建议都可以和博主分享啦,一起加油进步~