异常处理
异常:
在程序运行过程中,出现的不正常情况叫做异常
异常处理机制
条件:预先设置好对付异常的处理方法
过程:程序运行-->异常-->对异常进行处理-->处理完毕,程序继续运行-->
异常处理关键字:
try catch finally throw throws
特性:
1、相同的代码在运行的时候,根据输入的参数或者操作的不同,有可能会发生异常,有可能不会发生异常,所以应该在写代码的过程中尽可能的保证代码的正确性,不要到处都bug
2、如果要解决代码中出现的异常,需要添加非常复杂的代码逻辑来进行判断,会使代码变得非常臃肿,不利于维护,可读性比较差。(如使用if-else来解决异常,会很难处理所有的漏洞)因此,推荐大家使用异常机制来处理程序运行过程中出现的问题
3、程序在运行过程中如果出现了问题,会导致后面的代码无法正常执行,而使用异常机制之后,可以对异常情况进行处理,同时后续的代码会继续执行,不会中断整个程序
4、在异常的处理过程中,不要只是简单的输出错误,要尽可能的讲详细的异常信息进行输出
输出方式: e.printStackTrace(): 打印异常的堆栈信息,可以从异常信息的最后一行开始追踪,寻找自己编写的java类
不常用的输出方式:System.out.println(e.getMessage());//输出异常部分信息
代码执行可能发生的情况:
1、正常执行,只执行try中的代码
2、遇到异常情况,会处理try中异常代码之前的逻辑,后面的逻辑不会执行,最后会执行catch中的代码
3、使用多重catch的时候,会遇到异常子类不匹配的情况
此时依然会报错,因此建议在catch的最后将所有的异常的父类写上
如:catch(Exception e){
System.out.println("出现异常");
e.printStackTrace();
}
异常处理方式:
-
捕获异常
try{代码逻辑}catch(Exception e){异常处理逻辑}
实例:
try {
System.out.print("请输入被除数:");
int num1 = in.nextInt();
System.out.print("请输入除数:");
int num2 = in.nextInt();
System.out.println(String.format("%d / %d = %d",
num1, num2, num1 / num2));
}catch(Except e){
System.out.println("出现异常");
}
try{代码逻辑}catch(具体的异常Exception e){异常处理逻辑}catch(具体的异常):
实例:
try{
//代码逻辑
}catch(ArithmeticException e){
System.out.println("数学异常,除数不能是0");
e.printStackTrace();
}catch (InputMismatchException e){
System.out.println("输入的参数值类型不匹配");
e.printStackTrace();
}catch (NullPointerException e){
System.out.println("空指针异常");
e.printStackTrace();
}
该方式可以针对每一种具体的异常做相应的更丰富的处理
但当使用多重的catch的时候一定要注意相关异常的顺序,将子类放在最前面的catch,父类放在后面的catch
常见异常类型:
finally
在程序运行过程中,如果处理异常的部分包含finally的处理,那么无论代码是否发生异常,finally中的代码总会执行
finally包含哪些处理逻辑?
1、IO流的关闭操作一般设置在finally中
2、数据库的连接关闭操作设置在finally中
1、情况一(try中有return,finally中没有return):
public class TryTest{
public static void main(String[] args){
System.out.println(test());
}
private static int test(){
int num = 10;
try{
System.out.println("try");
return num += 80;
}catch(Exception e){
System.out.println("error");
}finally{
if (num > 20){
System.out.println("num>20 : " + num);
}
System.out.println("finally");
}
return num;
}
}
2、情况二(try和finally中均有return):
public class TryTest{
public static void main(String[] args){
System.out.println(test());
}
private static int test(){
int num = 10;
try{
System.out.println("try");
return num += 80;
}catch(Exception e){
System.out.println("error");
}finally{
if (num > 20){
System.out.println("num>20 : " + num);
}
System.out.println("finally");
num = 100;
return num;
}
}
}
3、情况三(finally中改变返回值num):
public class TryTest{
public static void main(String[] args){
System.out.println(test());
}
private static int test(){
int num = 10;
try{
System.out.println("try");
return num;
}catch(Exception e){
System.out.println("error");
}finally{
if (num > 20){
System.out.println("num>20 : " + num);
}
System.out.println("finally");
num = 100;
}
return num;
}
}
4、情况四(将num的值包装在Num类中):
public class TryTest{
public static void main(String[] args){
System.out.println(test().num);
}
private static Num test(){
Num number = new Num();
try{
System.out.println("try");
return number;
}catch(Exception e){
System.out.println("error");
}finally{
if (number.num > 20){
System.out.println("number.num>20 : " + number.num);
}
System.out.println("finally");
number.num = 100;
}
return number;
}
}
class Num{
public int num = 10;
}
1、
输出结果如下:
try
num>20 : 90
finally
90
分析:显然“return num += 80”被拆分成了“num = num+80”和“return num”两个语句,线执行try中的“num = num+80”语句,将其保存起来,在try中的”return num“执行前,先将finally中的语句执行完,而后再将90返回。
2、
输出结果如下:
try
num>20 : 90
finally
100
分析:try中的return语句同样被拆分了,finally中的return语句先于try中的return语句执行,因而try中的return被”覆盖“掉了,不再执行。
3、
输出结果如下:
try
finally
10
分析:虽然在finally中改变了返回值num,但因为finally中没有return该num的值,因此在执行完finally中的语句后,test()函数会得到try中返回的num的值,而try中的num的值依然是程序进入finally代码块前保留下来的值,因此得到的返回值为10。
4、
输出结果如下:
try
finally
100
从结果中可以看出,同样是在finally中改变了返回值num的值,在情况三中,并没有被try中的return返回(test()方法得到的不是100),但在这里却被try中的return语句返回了。
面试重点总结:
try语句在返回前,将其他所有的操作执行完,保留好要返回的值,而后转入执行finally中的语句,而后分为以下三种情况:
情况一:如果finally中有return语句,则会将try中的return语句”覆盖“掉,直接执行finally中的return语句,得到返回值,这样便无法得到try之前保留好的返回值。
情况二:如果finally中没有return语句,也没有改变要返回值,则执行完finally中的语句后,会接着执行try中的return语句,返回之前保留的值。
情况三:如果finally中没有return语句,但是改变了要返回的值,这里有点类似与引用传递和值传递的区别,分以下两种情况,:
1)如果return的数据是基本数据类型或文本字符串,则在finally中对该基本数据的改变不起作用,try中的return语句依然会返回进入finally块之前保留的值。
2)如果return的数据是引用数据类型,而在finally中对该引用数据类型的属性值的改变起作用,try中的return语句返回的就是在finally中改变后的该属性的值。
throws Exception 把异常往外扔
方法-->主方法-->JVM
throws:声明异常
在异常情况出现的时候,可以使用try...catch...finally的方式对异常进行处理,除此之外,可以将异常向外跑出,由外部的进行处理
1、在方法调用过程中,可以存在N多个方法之间的调用,此时假如每个方法中都包含了异常情况。
那么就需要在每个方法中都进行try。。catch,另外一种比较简单的方式,就是在方法的最外层调用处理一次即可
使用throws的方法,对所有执行过程中的所有方法出现的异常进行统一集中处理。
2、如何判断是使用throws还是使用try...catch..
最稳妥的方式是在每个方法中都进行异常的处理
偷懒的方式是判断在整个调用的过程中,外层的调用方法是否有对异常的处理,如果有,直接使用throws,如果没有,那么就要使用try...catch...
throw:抛出异常
自定义异常:
* 在java的api中提供了非常丰富的异常类,但是在某些情况下不太满足我们的需求,此时需要自定义异常
* 步骤:
* 1、继承Exception类
* 2、自定义实现构造方法
* 3、需要使用的时候,使用throw new 自定义异常的名称;
* 什么时候需要自定义异常?
* 一般情况下不需要
* 但是在公司要求明确,或者要求异常格式规范统一的时候是必须要自己实现的
实例://性别异常
public class GenderException extends Exception {
public GenderException(){
System.out.println("性别异常");
}
public GenderException(String msg){
System.out.println(msg);
}
}
//异常方法
public static void show() throws GenderException{
String gender = "1234";
if (gender.equals("man")){
System.out.println("man");
}else if(gender.equals("woman")){
System.out.println("woman");
}else{
// throw new Exception("性别出现错误");
throw new GenderException("gender is wrong");
}
}
//主方法
public static void main(String[] args) {
try {
show();
} catch (GenderException e) {
e.printStackTrace();
}