这里写目录标题
1 异常的概述和分类
- A:异常的概述
- 异常就是Java程序在运行过程中出现的错误。
- B:异常的分类
- 通过API查看Throwable
- Error
- 服务器宕机,数据库崩溃等
- Exception
C:异常的继承体系 - Throwable
- Error
- Exception
- RuntimeException
1.1 JVM默认是如何处理异常的
- A:JVM默认是如何处理异常的
- main函数收到这个问题时,有两种处理方式:
- a:
自己将该问题处理
,然后继续运行 - b:自己没有针对的处理方式,只有
交给调用main的jvm
来处理 - jvm有一个默认的异常处理机制,就将该异常进行处理.
- 并
将该异常的名称,异常的信息.异常出现的位置打印在了控制台上,同时将程序停止运行
- B:案例演示
- JVM默认如何处理异常
2 异常处理的两种方式
- try…catch…finally
- try catch
- try catch finally
- try finally
- throws
2.1 try…catch的方式处理异常1
当通过try catch将问题处理了,程序会继续向下执行。
- C:案例演示
- try…catch的方式处理1个异常
public static void main(String[] args) {
Demo2 d = new Demo2();
try{
int x = d.div(10, 0);
System.out.println(x);
}catch(ArithmeticException a) {
System.out.println("出错了,除数为零了");
}
System.out.println("1111111111111111");
}
2.2 try…catch的方式处理异常2
try后面如果跟多个catch,那么小的异常放前面,大的异常放后面,根据多态的原理,如果大的放前面,就会将所有的子类对象接收后面的catch就没有意义了。
- A:案例演示
- try…catch的方式处理多个异常
- JDK7以后处理多个异常的方式及注意事项
public static void main(String[] args) {
//demo1();
int a = 10;
int b = 0;
int[] arr = {11,22,33,44,55};
//JDK7如何处理多个异常
try {
System.out.println(a / b);
System.out.println(arr[10]);
} catch (ArithmeticException | ArrayIndexOutOfBoundsException e) {
System.out.println("出错了");
}
}
public static void demo1() {
int a = 10;
int b = 0;
int[] arr = {11,22,33,44,55};
try {
System.out.println(a / b);
System.out.println(arr[10]);
arr = null;
System.out.println(arr[0]);
} catch (ArithmeticException e) {
System.out.println("除数不能为零");
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println("索引越界了");
} catch (Exception e) { //Exception e = new NullPointerException();
System.out.println("出错了");
}
System.out.println("over");
}
2.3 编译期异常和运行期异常的区别
- A:编译期异常和运行期异常的区别
-
Java中的异常被分为两大类:编译时异常和运行时异常。
-
所有的
RuntimeException类及其子类的实例被称为运行时异常
,其他的异常就是编译时异常 -
编译时异常
- Java程序必须显示处理,否则程序就会发生错误,无法通过编译
-
运行时异常
- 无需显示处理,也可以和编译时异常一样处理
-
- B:案例演示
- 编译期异常和运行期异常的区别
public class Demo4_Exception {
/*
* 编译期异常和运行期异常的区别
编译时异常也叫做未雨绸缪异常(老师自己定义的)
未雨绸缪:在做某些事情的时候要做某些准备
* 编译时异常:在编译某个程序的时候,有可能会有这样那样的事情发生,比如
文件找不到,这样的异常就必须在编译的时候处理。
如果不处理编译通不过
* 运行时异常:就是程序员所犯得错误,需要回来修改代码
*/
public static void main(String[] args) {
try {
FileInputStream fis = new FileInputStream("xxx.txt");
} catch(Exception e) {
}
}
}
2.4 Throwable的几个常见方法
- A:Throwable的几个常见方法
- a:getMessage()
- 获取异常信息,返回字符串。
- b:toString()
- 获取异常类名和异常信息,返回字符串。
- c:printStackTrace()
- 获取异常类名和异常信息,以及异常出现在程序中的位置。返回值void。
- a:getMessage()
- B:案例演示
- Throwable的几个常见方法的基本使用
public static void main(String[] args) {
try {
System.out.println(1/0);
} catch (Exception e) { //Exception e = new ArithmeticException("/ by zero");
//System.out.println(e.getMessage()); //获取异常信息
//System.out.println(e); //调用toString方法,打印异常类名和异常信息
e.printStackTrace(); //jvm默认就用这种方式处理异常
}
}
2.5 throws的方式处理异常
- A:throws的方式处理异常
- 定义功能方法时,需要把出现的问题暴露出来让调用者去处理。
- 那么就通过throws在方法上标识。
- B:案例演示
- 举例分别演示编译时异常和运行时异常的抛出
public class Demo_Exception {
/**
* * A:throws的方式处理异常
* 定义功能方法时,需要把出现的问题暴露出来让调用者去处理。
* 那么就通过throws在方法上标识。
* B:案例演示
* 举例分别演示编译时异常和运行时异常的抛出
* 编译时异常的抛出必须对其进行处理
* 运行时异常的抛出可以处理也可以不处理
* @throws Exception
*/
public static void main(String[] args) throws Exception {
Person p = new Person();
p.setAge(-17);
System.out.println(p.getAge());
}
}
class Person {
private String name;
private int age;
public Person() {
super();
}
public Person(String name, int age) {
super();
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) throws AgeOutOfBoundsException {
if(age >0 && age <= 150) {
this.age = age;
}else {
//Exception e = new Exception("年龄非法");
//throw e;
throw new AgeOutOfBoundsException("年龄非法");
}
}
}
2.6 throw的概述以及和throws的区别
- A:throw的概述
- 在功能方法内部出现某种情况,程序不能继续运行,需要进行跳转时,就用throw把异常对象抛出。
- B:案例演示
- 分别演示编译时异常对象和运行时异常对象的抛出
- C:throws和throw的区别
- a:throws
- 用在方法声明后面,跟的是异常类名
- 可以跟多个异常类名,用逗号隔开
- 表示抛出异常,由该方法的调用者来处理
- b:throw
- 用在方法体内,跟的是异常对象名
- 只能抛出一个异常对象名
- 表示抛出异常,由方法体内的语句处理
- a:throws
2.7 finally关键字的特点及作用
- A:finally的特点
- 被finally控制的语句体一定会执行
- 特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))
- B:finally的作用
- 用于释放资源,在IO流操作和数据库操作中会见到
- C:案例演示
- finally关键字的特点及作用
public class Demo7_Finally {
/**
* C:案例演示
* finally关键字的特点及作用
*return语句相当于是方法的最后一口气,那么在他将死之前会看一看有没有finally帮其完成遗愿,如果有就 将finally执行
*后在彻底返回
*/
public static void main(String[] args) {
try {
System.out.println(10/0);
} catch (Exception e) {
System.out.println("除数为零了");
System.exit(0); //退出jvm虚拟机,这时就不会执行finally语句
return;
} finally {
System.out.println("看看我执行了吗");
}
}
}
2.8 异常 finally关键字的面试题
- A:面试题1
- final,finally和finalize的区别
- B:面试题2
- 如果catch里面有return语句,请问finally的代码还会执行吗? 如果会,请问是在return前还是return后。
package com.heima.test;
public class Test1 {
/**
* * A:面试题1
* final,finally和finalize的区别
* final可以修饰类,不能被继承
* 修饰方法,不能被重写
* 修饰变量,只能赋值一次
*
* finally是try语句中的一个语句体,不能单独使用,用来释放资源
*
* finalize是一个方法,当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
* B:面试题2
* 如果catch里面有return语句,请问finally的代码还会执行吗?如果会,请问是在return前还是return后。
*/
public static void main(String[] args) {
Demo d = new Demo();
System.out.println(d.method());
}
}
class Demo {
public int method() {
int x = 10;
try {
x = 20;
System.out.println(1/0);
return x;
} catch (Exception e) {
x = 30;
return x;
} finally {
x = 40;
//return x;不要在finally里面写返回语句,因为finally的作用是为了释放资源,是肯定会执行的
// 如果在这里面写返回语句,那么try和catch的结果都会被改变,所以这么写就是犯罪
}
}
}
3 自定义异常概述和基本使用
- A:为什么需要自定义异常
- 举例:人的年龄
- 好处,让我们更好的能够检索到自己的错误
- B:自定义异常概述
- 继承自Exception
- 继承自RuntimeException
- C:案例演示
- 自定义异常的基本使用
public class Demo_Exception {
public static void main(String[] args) {}
}
class AgeOutOfBoundsException extends Exception {
public AgeOutOfBoundsException() {//空参构造
super();
}
public AgeOutOfBoundsException(String message) {//有参构造
super(message);
}
}
3.1 异常的注意事项及如何使用异常处理
- A:异常注意事项
- a:子类重写父类方法时,子类的方法必须抛出相同的异常或父类异常的子类。(
父亲坏了,儿子不能比父亲更坏
) - b:如果父类抛出了多个异常,子类重写父类时,只能抛出相同的异常或者是他的
子集
,子类不能抛出父类没有的异常 - c:如果被
重写的方法没有异常抛出
,那么子类的方法绝对不可以抛出异常
,如果子类方法内有异常发生,那么子类只能try,不能throws
- a:子类重写父类方法时,子类的方法必须抛出相同的异常或父类异常的子类。(
- B:如何使用异常处理
-
原则:如果该功能内部可以将问题处理,用try,如果处理不了,交由调用者处理,这是用throws
-
区别:
- 后续程序需要继续运行就try
- 后续程序不需要继续运行就throws
-
如果JDK没有提供对应的异常,需要自定义异常。
-