java方法的理解、调用栈与异常处理
一、流程分支
If/else :基于boolean值的双分支
Switch:基于数字(整数、char、byte、枚举)、字符串
类型的多分支
Int month =5;
Switch
二、方法method
1.方法就是一个子程序
Java中方法的命名规范是驼峰命名法。
int add( int a, int b ){
int result = a + b;
return result;
}
float add( float a, float b){
float result = a + b;
return result;
}
2.方法也是有类型的
对于方法的使用者来说,返回值就是方法的类型
对于重载来说,方法签名就是其类型
方法签名:方法的名字+参数类型(多个,顺序严格的)
方法的返回值类型不属于签名的一部分
//这两个方法的方法签名相同,因此传的参数不知道调用哪一个方法,无法确定返回类型。
int add( int a, int b ){
int result = a + b;
return result;
}
float add( int a, int b ){
int result = a + b;
return result;
}
签名不同才能重载!!
Tips:
1>对于java8中的拉姆达表达式来说,方法的类型包括
返回值类型->(参数类型<多个,顺序严格的>)
2>方法的返回值有一个特殊的类型void,、
没有返回值,也就是方法中没有return语句
3>方法的参数传递时有2种传值形式,实际是由参数的类型的性质决定的。
基本类型:按值传递,把值复制到方法中
引用类型:只是传递对象的引用,如果在方法中改变了对象(内部属性),就会影响这个对象
4>方法(包括面向对象)都不是必须的!方法和面向对象都是给程序设计人员使用的。所以写代码时,如果要定义方法,就是站在设计师的高度去完成方法的设计。
3.设计的必要性:
设计师有必要的(有利于提升质量、可维护性、效率、重用),但是设计是无止境的,适可而止。评价设计优劣的标准:简化、不用重复(消除重复代码)
4.提取方法,将功能只能分散
如果一个方法代码太多,指责太多,则表示需要对这个方法进行重构(Rafactor)。通常会使用提取方法的功能智能进行分散
IDE(Ecllipse)对此提供非常好的支持
举例:单位转换程序,所以的代码都可以写在main方法中。但是将调度职责和具体的转换职责进行拆分,将会使程序结构更加清晰。整个程序也会更加简单。
public Class UnitClac{
//交互逻辑
static void main(String[] args){
//进行转换调度
}
//业务逻辑
static void c2f(Scannerscanner){
//将摄氏度转换为华氏度
}
static void f2c(Scannerscanner){
//将华氏度转换为摄氏度
}
}
任何应用都由应用交互逻辑和业务逻辑2部分构成。
交互逻辑可以随意改变,而且经常会随技术和流行趋势变化。
业务逻辑相对稳定。
交互逻辑通常围绕UI展开。
业务员逻辑通常围绕数据展开
具体来说就是收集数据、存储、分析、展示数据。
Static (静态的)方法没有充分体现面向对象的特征。
对于static方法来说,类只是一个盒子(容器),static方法与对象(实例)关系不大。
三、程序调试
通过调试可以观察程序的执行过程和内部数据
调试是一个非常强大的能力(断点和单步执行是由CPU和JVM联合支持的)
通过“虫子”图标可以进入调试模式,如果没有看到调试视图,可以在右上角进行透视图的切换
加断点:让程序停留在断点的位置
单步执行:F5/F6
F5:进入方法
F6:越过方法
四、调用栈
先入后出
栈是一个只有一个口的容器,先进入栈的会落到栈底,出栈的时候最后出。最后进入栈的,在栈顶,出栈时先出。
方法调用时,需要在内存中开辟一块存储空间做为线程栈空间
每个线程都由自己的栈
调用方法时,会在栈中压入一个栈帧,用来存储这个方法的参数和局部变量
方法返回时 ,栈帧就会弹出,方法的参数和局部变量就会清除
方法调用时,调用栈不断处于涨落之中
如果调用的层级过深,调用栈可能会溢出
因为代码执行的速度很快,所以栈帧的生存时间很短,瞬间生灭
所以局部变量无法被外部使用
五、异常处理
程序执行时,遇到错误(调用栈中的错误)就会停止执行,如果错误数据不能清除掉,程序就无法恢复,最终崩溃。
而java等现代编程语言普遍提高了清理错误数据的机制
异常处理
早期的c语言没有异常处理机制
在java中,有一类异常很特别,它们叫做受查异常
受查异常必须传递出去(thows)或者处理掉(try/catch),不能不管
try(){
}
当代码执行到try时,会建立一个安全点,一旦在try中发生错误
JVM就会检查并收集错误信息(错误的原因、错误的位置、当前调用栈的结构等)
然后利用这些信息创建一个Exception对象(类型取决于错误原因)
然后抛出(throw)这个异常对象
异常对象的传播当异常被抛出throw之后,JVM会沿着调用栈从上往下逐帧查找try建立的安全点,直到找到一个符合条件的catch或者到达栈底(崩溃)
Catch (SomeTypeException ex){
//清理
//异常对象的使用
}
//从这里继续执行
异常的catch
Catch是一个匹配的过程,只有类型匹配成功,才会接受(捕获),否则继续沿着调用栈查找。
当异常被处理后,上面的栈帧就会被清除
如果在错误之前打开了一个系统资源,则该资源就有可能没有正确地关闭,所有try catch后面可以写finally{ }
finally一定会被调用,可以用来关闭资源。
异常对象中包含的信息可以通过异常对象提供的方法获取到
getMessage()
getStackTrace()
printStackTracd() 可以把异常信息打印到控制台或指定的输出流中(保存成文件等)
作业:
1.画一个调用栈来表示多层方法调用的过程
static void main(){
double a = 3;
double b =5;
double c = calcArc(3,5)
}
static double calcArc(double x, double y){
double m = pow(x);-+
double n = pow(y);
double 0 = m+n;
return sqrt(o);
}
2.Square类 a边长
Rectangle类 a,b长宽
Circle 类 r半径
Sidelength()周长
3.画一个异常传播的示意图