标签:代码片段 应该 知识 first 相关 意义 array except 数组
在程序开发中,异常处理也是我们经常使用到的模块,只是平常很少去深究异常模块的一些知识点。比如,try-catch 处理要遵循的原则是什么,finally 为什么总是能执行,try-catch 为什么比较消耗程序的执行性能等问题,我们本讲内容都会给出相应的答案,当然还有面试中经常被问到的异常模块的一些面试题,也是我们本篇要讲解的重点内容。
先来看看异常处理的语法格式:
try{ ... } catch(Exception e){ ... } finally{ ... }
其中,
异常处理的基本使用,具体可以参考下面的代码段:
try {
int i = 10 / 0;
} catch (ArithmeticException e) {
System.out.println(e);
} finally {
System.out.println("finally");
}
多 catch 的使用,具体可以参考下面的代码段:
try {
int i = Integer.parseInt(null);
} catch (ArithmeticException ae) {
System.out.println("ArithmeticException");
} catch (NullPointerException ne) {
System.out.println("NullPointerException");
} catch (Exception e) {
System.out.println("Exception");
}
需要注意的是 Java 虚拟机会从上往下匹配错误类型,因此前面的 catch 异常类型不能包含后面的异常类型。比如上面的代码如果把 Exception 放在最前面编译器就会报错,具体可以参考下面的图片。
随着 Java 语言的发展,JDK 7 的时候引入了一些更加便利的特性,用来更方便的处理异常信息,如 try-with-resources 和 multiple catch,具体可以参考下面的代码段:
try (FileReader fileReader = new FileReader("");
FileWriter fileWriter = new FileWriter("")) { // try-with-resources
System.out.println("try");
} catch (IOException | NullPointerException e) { // multiple catch
System.out.println(e);
}
先来看下面这段代码,有没有发现一些问题?
try {
// ...
int i = Integer.parseInt(null);
} catch (Exception e) {
}
以上的这段代码,看似“正常”,却违背了异常处理的两个基本原则:
异常处理固然好用,但一定不要滥用,比如下面的代码片段:
// 使用 com.alibaba.fastjson
JSONArray array = new JSONArray();
String jsonStr = "{'name':'laowang'}";
try {
array = JSONArray.parseArray(jsonStr);
} catch (Exception e) {
array.add(JSONObject.parse(jsonStr));
}
System.out.println(array.size());
这段代码是借助了 try-catch 去处理程序的业务逻辑,通常是不可取的,原因包括下列两个方面。
以上使用 try-catch 处理业务的代码,可以修改为下列代码:
// 使用 com.alibaba.fastjson
JSONArray array = new JSONArray();
String jsonStr = "{'name':'laowang'}";
if (null != jsonStr && !jsonStr.equals("")) {
String firstChar = jsonStr.substring(0, 1);
if (firstChar.equals("{")) {
array.add(JSONObject.parse(jsonStr));
} else if (firstChar.equals("[")) {
array = JSONArray.parseArray(jsonStr);
}
}
System.out.println(array.size());
答:try 不能单独使用,否则就失去了 try 的意义和价值。
try {
int i = 10 / 0;
} catch {
System.out.println("last");
}
答:不能正常运行,catch 后必须包含异常信息,如 catch (Exception e)。
try {
int i = 10 / 0;
} finally {
System.out.println("last");
}
答:可以正常运行。
try {
int i = 10 / 0;
System.out.println("try");
} catch (Exception e) {
int j = 2 / 0;
System.out.println("catch");
} finally {
System.out.println("finally");
}
System.out.println("main");
答:程序会打印出 finally 之后抛出异常并终止运行。
try {
System.out.println("try");
} catch (Exception e) {
System.out.println("catch");
} finally {
int k = 3 / 0;
System.out.println("finally");
}
System.out.println("main");
答:程序在输出 try 之后抛出异常并终止运行,不会再执行 finally 异常之后的代码。
答:常见的运行时异常如下:
试图通过字符串来加载某个类时引发的异常;
答:Exception 和 Error 都属于 Throwable 的子类,在 Java 中只有 Throwable 及其之类才能被捕获或抛出,它们的区别如下:
答:它们的区别如下:
答:Integer.parseInt(null) 和 Double.parseDouble(null) 抛出的异常类型不一样,如下所示:
至于为什么会产生不同的异常,其实没有特殊的原因,主要是由于这两个功能是不同人开发的,因而就产生了两种不同的异常信息。
答:这个问题要从 JVM(Java 虚拟机)层面找答案了。首先 Java 虚拟机在构造异常实例的时候需要生成该异常的栈轨迹,这个操作会逐一访问当前线程的栈帧,并且记录下各种调试信息,包括栈帧所指向方法的名字,方法所在的类名、文件名,以及在代码中的第几行触发该异常等信息,这就是使用异常捕获耗时的主要原因了。
答:常见的 OOM 原因有以下几个:
public static int getNumber() {
try {
int number = 0 / 1;
return 2;
} finally {
return 3;
}
}
A:0
B:2
C:3
D:1
答:3
题目解析:程序最后一定会执行 finally 里的代码,会把之前的结果覆盖为 3。
答:finally、finalize 的区别如下:
答:finally 总会被执行,都是编译器的作用,因为编译器在编译 Java 代码时,会复制 finally 代码块的内容,然后分别放在 try-catch 代码块所有的正常执行路径及异常执行路径的出口中,这样 finally 才会不管发生什么情况都会执行。
欢迎关注我的公众号,回复关键字“Java” ,将会有大礼相送!!! 祝各位面试成功!!!
标签:代码片段 应该 知识 first 相关 意义 array except 数组
原文地址:https://www.cnblogs.com/dailyprogrammer/p/12267585.html