java中,将程序执行过程中发生的不正常行为称为异常,java中不同类型的异常,都有与其对应的类来进行描述
public class TextDame {
public static void main(String[] args) {
int[] array = null;
System.out.println(array.length);
}
}
public class TextDame {
public static void main(String[] args) {
int[] array = {1,2,3,4};
System.out.println(array[10]);
}
}
public class TextDame {
public static void main(String[] args) {
int a = 10 / 0;
System.out.println(a);
}
}
在程序编译期间发生的异常,称为编译时异常,也称为受检查异常,ClassNotFoundException和CloneNotSupportedException以及IOException及其子类都是编译时异常
在程序执行期间发生的异常,称为运行时异常,也称为非受检查异常,RunTimeException以及其子类对应的异常,都称为运行时异常
(1)事前防御型:在操作之前就做充分的检查,使用大量的if else语句进行判断,缺点是正常流程和错误处理流程代码混在一起,代码整体显的比较混乱
(2)事后认错型:使用try catch语句,优点是正常流程和错误流程是分离开的, 程序员更关注正常流程,代码更清晰,容易理解代码,异常处理主要的5个关键字:throw、try、catch、final、throws
借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者
注意:
public class TextDame {
public static void main(String[] args) {
int[] array = null;
try {
if (array == null){
throw new NullPointerException("空指针异常");
}
}catch (NullPointerException e){
System.out.println("捕获到空指针异常");
}
}
}
在方法声明时参数列表之后,当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛给方法的调用者来处理
注意:
class Person implements Cloneable {
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
public class TextDame {
public static void main(String[] args) throws CloneNotSupportedException {
Person person = new Person();
Person person1 = (Person) person.clone();
}
}
throws对异常并没有真正处理,而是将异常报告给抛出异常方法的调用者,由调用者处理。如果真正要对异常进行处理,就需要try-catch
语法格式:
try{
// 存放可能出现异常的代码
}catch(要捕获的异常类型 e){
// 如果try中的代码抛出异常了,此处catch捕获时异常类型与try中抛出的异常类型一致时,或者是try中抛出异常的父类时,就会被捕获到,对异常就可以正常处理,处理完成后,跳出try-catch结构,继续执行后序代码
}finally{
// 此处代码一定会被执行到
}
public class TextDame {
public static void main(String[] args) {
try {
int[] array = null;
System.out.println(array.length);
}catch (NullPointerException e){
System.out.println("捕获到空指针异常");
}
}
}
注意:
public class TextDame {
public static void main(String[] args) {
try {
int[] array = null;
System.out.println(array.length);
}catch (ArrayIndexOutOfBoundsException e){
System.out.println("捕获到空指针异常");
}
}
}
(1)在写程序时,有些特定的代码,不论程序是否发生异常,都需要执行,比如资源回收,因为异常会引发程序的跳转,可能导致有些语句执行不到,finally就是用来解决这个问题的,finally中的代码一定会执行的
(2)finally 执行的时机是在方法返回之前(try 或者 catch 中如果有 return 会在这个 return 之前执行 finally),但是如果finally 中也存在 return 语句,那么就会执行 finally 中的 return, 从而不会执行到try中原有的 return
(1)如果本方法中没有合适的处理异常的方式,就会沿着调用栈向上传递
(2)如果向上一直传递都没有合适的方法处理异常,最终就会交给 JVM 处理,程序就会异常终止,和我们最开始未使用 try catch 时是一样的
总结:
具体方式:
//自定义异常
class MyException extends RuntimeException{
public MyException() {
super();
}
public MyException(String s) {
super(s);
}
}
//编译时异常
class MyException2 extends Exception{
public MyException2() {
super();
}
public MyException2(String s) {
super(s);
}
}
public class TextDame {
public static void function1(){
int a = 10;
if(a == 10) {
throw new MyException("自定义异常");
}
}
public static void function2() throws MyException2 {
int a = 10;
if(a == 10) {
throw new MyException2("自定义异常");
}
}
public static void function3() {
try {
int a = 10 ;
if(a == 10) {
throw new MyException2("哈哈 我自己写的异常!");
}
}catch (MyException2 e) {
}
}
}