标签:log 异常类 强制 lis 抛出异常 判断 name 应该 控制
往期回顾:上一篇我们大致总结了异常的继承体系,说明了Exception和Error两个大类都继承于顶级父类Throwable,又谈到编译时异常与运行时异常的区别,谈到异常的处理方式,以及处理方式中关于捕获方式的几种类型。
本篇承上启下,将从异常的其余部分进行总结,但是毕竟现在处于初学阶段,未必能够体会异常在真实场景中运用的便利之处,所以本文只是对目前所学内容的归纳整理,后续新的体会将会及时更新。
- 从当前方法开始,沿着方法的调用链,按照异常的反向传播方向找到异常的处理代码。
- 从第一个到最后一个检查catch块,判断是否相匹配。如果是,那么恭喜!直接进入catch块中执行处理异常语句;如果不是,就将该异常传给方法的调用者,在调用者中继续执行相同步骤:匹配就处理,不匹配就向上传……
- 直到最后如果都没有找到的话,程序将会终止,并在打印台上打印出错信息。
如果是相对单一方法而言,其实是很简单的;如果方法层层嵌套呢,情况又是咋样的呢,咱们来验证一下以上内容:
//主方法
public static void main(String[] args) {
try{
//调用m1()
m1();
System.out.println("ExceptionMyDemo.main");
}catch (Exception e){
System.out.println("ExceptionMyDemo.main.catch");
}
}
//m1()
private static void m1(){
try{
//调用m2()
m2();
System.out.println("ExceptionMyDemo.m1");
}catch (NullPointerException e){
System.out.println("ExceptionMyDemo.m1.catch");
}
}
//m2()
private static void m2(){
String str = null;
System.out.println(str.hashCode());
}
//测试结果:
ExceptionMyDemo.m1.catch
ExceptionMyDemo.main
catch (ArithmeticException e)
,这时的测试结果会是这个样子://更改之后的测试结果:
ExceptionMyDemo.main.catch
因为m2()抛出的异常在m1中并没有被合适地处理,所以向上抛出,在main方法中找到了处理方法,遂执行处理语句。
因为抛出的异常没人处理,它就会在控制台上打印异常的栈轨迹,关于抛出的异常信息,我们接下来进行详细分析。
我们提到,无论是虚拟机抛出异常还是我们主动抛出,异常的错误信息都包含其中,以便于我们得知并更好地处理异常,那么顺着上面所说,我们刚刚看到的就是异常的栈轨迹:
public void printStackTrace()
:默认将该Throwable对象及其调用栈的跟踪信息打印到标准错误流。public String getMessage()
:返回描述异常对象信息的字符串。public String toString()
:异常信息message为空就返回异常类的全名,否则返回全名:message
的形式。public StackTraceElement[] getStackTrace()
:返回栈跟踪元素的数组,表示和该异常对象相关的栈的跟踪信息。名言警句:无论异常是否会发生,finally修饰的子句总是会被执行。
于是我们进行了简单的尝试:
public static void m2(){
try{
System.out.println("0");
System.out.println(1/0);
System.out.println("1");
}
catch (Exception e){
System.out.println("2");
}
finally {
System.out.println("3");
}
System.out.println("4");
}
//测试结果
0 2 3 4 被打印在控制台上
System.out.println(1/0);
时发生了异常,于是进入catch块,finally子句必会被执行,然后执行try语句后的下一条语句。想象以下:假如把接收异常的实例类型改为另外一个不匹配的类型的话,也就是说无法正常捕获,结果又会如何呢?结果如下:
还有一个注意点就是4也没有被打印出来,是因为没有捕获到异常,将会把异常抛给调用者,所以不会执行System.out.println("4");
。
但是,化名为几千万个为什么的我又开始疑惑了,我们直到return可以将方法直接返回,强制退出。那么如果在try中使用return语句,finally还会不会不忘初心,继续执行呢?
前方高能!各单位注意!!!
猜猜看,这四个方法执行结果是啥呢?
private static int m1(){
try{
return 1;
}catch(Exception e){
}
return 2;
}
private static int m2(){
try{
return 1;
}finally {
return 2;
}
//使用finally子句时可以省略catch块
}
private static int m3(){
try{
return 1;
}finally {
try{
return 2;
}finally {
return 3;
}
}
}
private static int m4(){
int i = 4;
try{
return i++;
}finally {
i++;
}
}
答案揭晓:分别是:1,2,3,4。你们猜对了吗?哈哈……
我想前三个答案应该是毋庸置疑的,但是这第四个就有点离谱了。不是说finally语句一定会执行吗,执行哪去了呢,你要是执行的话,你i难道不应该变成6了吗?
额……咳咳,这个嘛,我也有点迷惑,但是经过一番讨教,稍微懂了一些:
但是如果进行改变的是引用数据类型的变量时,那么就会随之改变了,人家村的是地址,改的就是本身。我在这边就稍微来个简单的例子奥:
public static Student m(){
Student s = new Student();
try{
s.age = 20;
s.name = "天乔";
return s;
}finally {
s.name = "巴夏";
s.age = 2;
}
}
//测试结果
//Student{age=2, name='巴夏'}
本文若有叙述不当之处,还望评论区批评指正哦!
标签:log 异常类 强制 lis 抛出异常 判断 name 应该 控制
原文地址:https://www.cnblogs.com/summerday152/p/12180854.html