异常.png
JVM 的默认处理方案
如果程序出现了问题,我们没有做任何处理,最终 JVM 会做默认的处理
- 把异常的名称,异常的原因及异常出现的位置等信息输出在控制台
- 程序停止执行
异常处理
如果程序出现了问题,我们需要自己来处理,有两种方案:
try ... catch ...
throws
异常处理之 try...catch...
- 格式
try {
可能出现的异常代码;
} catch(异常类名 变量名) {
异常的处理代码;
}
- 执行流程:
程序从 try 里面的代码开始执行
出现异常,会自动生成一个异常类对象,该异常类对象将被提交给 java 运行时系统
当 java 运行时系统接收到异常对象时,会到 catch 中去找匹配的异常类,找到后进行异常的处理
执行完毕后,程序还可以继续往下执行
public static void main(String[] args){
System.out.println("开始");
method();
System.out.println("结束");
}
public static void method() {
try {
int[] arr = {1, 2, 3};
System.out.println(arr[3]);
} catch (ArrayIndexOutOfBoundsException e){ // new ArrayIndexOutOfBoundsException();
// System.out.println("你访问的数组索引不存在");
e.printStackTrace();
}
}
Throwable 的成员方法
Throwable的成员方法.png
public static void main(String[] args){
System.out.println("开始");
method();
System.out.println("结束");
}
public static void method() {
try {
int[] arr = {1, 2, 3};
System.out.println(arr[3]);
} catch (ArrayIndexOutOfBoundsException e){ // new ArrayIndexOutOfBoundsException();
// System.out.println("你访问的数组索引不存在");
// e.printStackTrace();
System.out.println(e.getMessage());
System.out.println(e.toString());
e.printStackTrace(); // 最全
}
}
编译时异常和运行时异常的区别
java 中的异常被分为两大类: 编译时异常和运行时异常,也被成为受检异常和非受检异常
所有的 RuntimeException 类及其子类被称为运行时异常,其他异常都是编译时异常
- 编译时异常: 必须显示处理,否则程序就会发生错误,无法通过编译
- 运行时异常: 无需显示处理,也可以和编译时异常一样处理
public static void main(String[] args){
// method();
method2();
}
// 编译时异常
public static void method2() {
try{
String s = "2020-02-09";
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date d = sdf.parse(s);
System.out.println(d);
} catch (ParseException e) {
e.printStackTrace();
}
}
// 运行时异常
public static void method() {
try {
int[] arr = {1, 2, 3};
System.out.println(arr[3]); // ArrayIndexOutOfBoundsException
} catch (ArrayIndexOutOfBoundsException e) {
e.printStackTrace();
}
}
异常处理之 throws
虽然我们通过 try...catch... 可以对异常进行处理,但是并不是所有的情况我们都有权限进行异常的处理
也就是说,有些时候可能出现的异常是我们处理不了的,这个时候该怎么办?
针对这种情况,java 提供了 throws 的处理方案
- 格式
throws 异常类名;
注意: 这个格式是跟在方法的括号后面的
- 编译时异常必须要处理,两种处理方案:
try...catch...
或者throws
,如果采用throws
这种方案,将来谁调用谁处理 - 运行时异常可以不处理,出现问题后,需要我们回来修改代码
public static void main(String[] args){
System.out.println("开始");
// method();
try {
method2();
} catch (ParseException e) {
e.printStackTrace();
}
System.out.println("结束");
}
// 编译时异常
public static void method2() throws ParseException {
String s = "2020-02-09";
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date d = sdf.parse(s);
System.out.println(d);
}
// 运行时异常
public static void method() throws ArrayIndexOutOfBoundsException{
int[] arr = {1, 2, 3};
System.out.println(arr[3]); // ArrayIndexOutOfBoundsException
}
自定义异常
- 格式:
public class 异常类名 extends Exception {
无参构造
带参构造
}
- 范例:
public class ScoreException extends Exception {
public ScoreException() {}
public ScoreException(String message) {
super(message);
}
}
public class ScoreException extends Exception {
public ScoreException() {}
public ScoreException(String message) {
super(message);
}
}
public class Teacher {
public void checkScore(int score) throws ScoreException {
if (score<0 || score>100) {
throw new ScoreException("你给的分数有误,应该在0-100之间");
} else {
System.out.println("分数正常");
}
}
}
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
System.out.println("请输入分数: ");
int score = sc.nextInt();
Teacher t = new Teacher();
try {
t.checkScore(score);
} catch (ScoreException e) {
e.printStackTrace();
}
}
throws 和 throw 的区别
- throws
- 用在方法声明后面,跟的是异常类名
- 表示抛出异常,由该方法的调用者来处理
- 表示出现异常的一种可能性,并不一定会发生这些异常
- throw
- 用在方法体内,跟的是异常对象名
- 表示抛出异常,由方法体内的语句处理
- 执行throw 一定抛出了某种异常