异常的概念

异常:
指的是程序执行过程中,出现的非正常的情况,最终会导致JVM的非正常停止
在Java等面向对象的编程语言中,异常本身是一个类,,产生异常就是创建异常对象并抛出一个异常对象。java处理异常的方式是中断
注意:异常并不是语法错误

异常体系

异常机制其实就是帮助我们找到程序中的问题,异常的根类:java.lang.Throwable

它有两个子类
子类1:
java.lang.Error
Error不能处理的,只能避免
子类2:
java.lang.Exception
Exception使用不当导致,可以避免的

异常的分类

java.lang.Throwable类是 Java语言中是所有错误或异常的超类
Exception:编译期异常,进行编译(写代码)java程序出现问题

RuntimeException:运行期异常,java程序运行过程中出现的异常
相当于程序得了一个小毛病(感冒,发烧),把异常处理掉(打点滴,吃点药),程序可以继续执行

Error: 错误
错误:相当于程序得了一个无法治愈的毛病。我们就必须修改源代码,程序才能继续运行

异常处理的过程解析

1.当JVM检测出程序中出现异常后,首先会执行两件事情

1.JVM 会根据异常产生的原因创建一个异常对象,这个异常包含了异常产生的(内容,原因,位置)

2.如果异常是在方法体中,方法体没有try…catch ,那么JVM就会把异常对象抛出给方法的调用者main方法来处理这个异常

2.当main方法接收到这个异常对象,main方法也没有异常处理逻辑,所有继续把对象抛出给main方法的调用者JVM处理

3.JVM接收到这个异常对象,做了两件事情

1.把异常对象(内容,原因,位置)以红色的字体打印在控制台
2.JVM会终止当前正在执行的java程序 -》 中断处理

异常的处理

java异常处理的五个关键字:try catch finally throw throws

抛出异常:throw
作用:
可以使用throw关键字,在指定的方法中抛出指定的异常

使用格式:
throw new XXXException(“异常产生的原因”);

注意事项:
1.throw关键字必须写在方法的内部
2.throw关键字后边new的对象必须是Exception 或者是Exception的子类对象
3.throw关键字抛出指定的异常对象,我们就必须处理这个异常对象

两种处理方式:

1.throw关键字后边创建的是RuntimeException或者是RuntimeException的子类我们可以不处理,默认交给JVM来处理(打印异常对象,中断程序)

2.throw关键字后边创建的是编译异常(写代码的时候报错),我们就必须处理这个异常,要么thows 要么try…catch

以后在工作中,我们首先必须对方法传递过来的参数进行合法性校验
如果参数不合法,我们就必须使用抛出异常的方式,告知方法的调用者,传入的参数有问题

我们可以对传递过来的参数 数组,进行合法性校验
如果数组arr的值是null
那么我们就抛出空指针异常,告知方法的调用者“传递的数组的值是空”
注意:NullPointerException是一个运行期异常。我们不用处理,默认交给JVM处理

if (arr == null) {
	throw new NullPointerException("传递的数组的值是空");
}

我们对传递过来的参数 索引 进行合法性校验
如果index的范围大于数组的长度
我们就抛出数组索引越界异常,告知方法的调用者“传入的索引超出了数组的使用范围”
ArrayIndexOutOfBoundsException是运行期异常,我们无需处理。默认交给JVM处理

if (index<0 || index>=arr.length) {
	throw new ArrayIndexOutOfBoundsException("传入的索引超出了数组的使用范围");
}
public static int method(int[] arr,int index) {
		/*
		 * 我们可以对传递过来的参数 数组,进行合法性校验
		 * 如果数组arr的值是null
		 * 那么我们就抛出空指针异常,告知方法的调用者“传递的数组的值是空”
		 * 注意:NullPointerException是一个运行期异常。我们不用处理,默认交给JVM处理
		 * */
		if (arr == null) {
			throw new NullPointerException("传递的数组的值是空");
		}
		/*
		 * 我们对传递过来的参数 索引 进行合法性校验
		 * 如果index的范围大于数组的长度
		 * 我们就抛出数组索引越界异常,告知方法的调用者“传入的索引超出了数组的使用范围”
		 * ArrayIndexOutOfBoundsException是运行期异常,我们无需处理。默认交给JVM处理
		 * */	
		if (index<0 || index>=arr.length) {
			throw new ArrayIndexOutOfBoundsException("传入的索引超出了数组的使用范围");
		}
		return arr[index];
	}

PS补充一个Objects非空判断的方法

Objects类中有一些静态的实用方法组成。
这些方法有空指针安全的方法(null - save)
有空指针不安全的方法(null - tolerant)容忍空指针的

public static T requireNonNUll)(T obj) 查看指定对象是否为null

查看指定对象是否为null
源码:

public static<T> T requireNonNUll)(T obj){
 	if(obj == bull){
 		throw new NullPointerException();
 		}
 	return obj;
 }
public class Demo04Objects {
	public static void main(String[] args) {
		menthod(null);
	}
	public static void menthod(Object obj) {
		/*if(obj==null) {
			throw new NullPointerException("空指针异常");
		}*/
		
		//等效方法:
		Objects.requireNonNull(obj);//java.lang.NullPointerException
		//重构方法
		Objects.requireNonNull(obj,"传入的对象为null");// java.lang.NullPointerException: 传入的对象为null
	}
}

异常处理的第一种方法:声明异常 : throws

异常处理的第一种方式,交给别人处理
作用:
当方法内部抛出异常对象的时候,那么我们就必须处理这个异常对象
可以使用throws 关键字处理异常对象,会把异常对象声明抛出给方法的调用者处理(自己不处理,给别人处理),最终交给JVM处理–》中断处理

使用格式:在方法声明时使用

修饰符 返回值类型  方法名(参数列表) throws AAAException,BBBException,CCCException{
 		throw new  AAAException(“产生原因”);
 		throw new  BBBException(“产生原因”);
 		throw new  CCCException(“产生原因”);
 		....
 	}

注意:

1.throws关键字必须写在方法声明出

2.throws关键字后边声明的异常必须是Exception 和Exception的子类

3.方法名内部如果抛出了多个异常,那么throws后边必须也声明多个异常
如果抛出的多个异常对象有 子父类关系,那么直接声明父类异常即可

4.调用了一个声明抛出异常的方法,我们就必须的处理声明的异常
要么继续使用throws声明抛出 交给方法的调用者处理,最终交给Jvm处理
要么try…catch 自己处理异常

public class Demo05Throws{
	public static void main(String[] args) throws Exception{
		//FileNotFoundException extends IOException extends Exception
		//所以可以直接只写一个父类
		readFile("c:\\a.te");
	}
	/*
	 * 	定义一个方法:对传递的文件路径进行合法性判断
	 * 	如果路径不是“c:\\a.txt”,那么我们就抛出文件找不到异常对象,告知方法的调用者
	 * 	注意:FileNotFoundException 是编译异常,抛出了编译异常,就必须处理这个异常
	 * 	可以使用使用throws声明继续抛出FileNotFoundException这个异常  交给方法的调用者处理
	 * */	
	public static void  readFile(String fileName) throws FileNotFoundException,IOException{
		if (!fileName.equals("c:\\a.txt")) {
			throw new  FileNotFoundException("传递的文件路径不是:c:\\a.txt");
		}
		//如果传递的路径,不是.txt结尾 ,那么我们就抛出IO异常对象	
		if (!fileName.endsWith(".txt")) {
			throw new IOException("文件后缀有误");//Exception in thread "main" java.io.IOException: 文件后缀有误
		}
		System.out.println("路径正常,读取文件!");
	}
}

异常处理的第二种方法:捕获异常 try…catch

try…catch 异常处理的第二种方式,自己处理异常

格式:

try{
		可能要产生异常的代码		
}catch (定义一个异常的变量,用来接收try 中抛出的异常对象){

异常的处理逻辑,产生异常对象之后,怎么处理异常对象。
一般在工作中,会把异常的信息记录到一个日志中

}
 	...
 catch(异常类名  变量名){
 
 	}

注意:
1.try中可能会抛出多个异常,那么就可以使用多个catch来处理这些异常对象
2.如果try中产生了 异常,那么就会执行catch中异常处理逻辑,执行完毕catch中的处理逻辑,继续执行try…catch之后的代码
如果try中没有产生异常,那么就不会执行catch中异常的处理逻辑,执行完try中的代码,继续执行try…catch之后的代码

public class Demo06Trycatch {
	public static void main(String[] args){
		try {
			readFile("c.//a.txt"); //产生异常的代码
		} catch (IOException e) { //try 中抛出什么异常对象,catch就定义什么异常变量,用来接收这个异常对象
			//异常的处理逻辑,产生异常对象之后,怎么处理异常对象。
			System.out.println("传递的文件后缀不是.txt");
		}
		System.out.println("后续代码");
	}
	public static void  readFile(String fileName) throws IOException {
		//如果传递的路径,不是.txt结尾 ,那么我们就抛出IO异常对象	
		if (!fileName.endsWith(".txt")) {
			throw new IOException("文件后缀有误");//Exception in thread "main" java.io.IOException: 文件后缀有误
		}
		System.out.println("路径正常,读取文件!");
	}
}

Throwable类中3个处理异常的方式

String getMessage() 获取异常的描简短述信息,原因(提示给用户的时候,就提示错误原因)

String toString() 获取异常的类型和异常的详细描述信息

void printStackTrace() JVM打印异常对象,默认此方法,打印的异常信息是最全的

System.out.println(e.getMessage()); //文件后缀有误
System.out.println(e.toString());//java.io.IOException: 文件后缀有误
e.printStackTrace(); //打印异常出现的原因,异常出现的内容,异常出现的位置

finally代码块

finally 有一些特定的代码无论异常是否发生,都需要执行。
另外,因为异常会引发程序跳转,导致有些语句执行不到,而finally就是解决这个问题的。finally代码块中的代码是一定会执行的

格式:

try{
  			可能要产生异常的代码
 		}catch (定义一个异常的变量,用来接收try 中抛出的异常对象){
 			异常的处理逻辑,产生异常对象之后,怎么处理异常对象。
  			一般在工作中,会把异常的信息记录到一个日志中
  		}
  		...
 		catch(异常类名  变量名){
 
 		}finally{
 			//无论是否出现异常,都会执行
	}

注意:
1.finally不能单独使用,必须和try一起使用
2.finally一般用于资源释放(资源回收),无论程序是否出现异常,最后都要资源释放(IO)

public class Demo08finally {
	public static void main(String[] args) {
		try {
			//可能会产生异常的代码
			readFile("c://a.tx");
		} catch (IOException e) {
			//异常的处理逻辑
			e.printStackTrace();
		}finally {
			//无论是否出现异常,都会执行
			System.out.println("资源释放");
		}
	}
	public static void  readFile(String fileName) throws IOException {
		//如果传递的路径,不是.txt结尾 ,那么我们就抛出IO异常对象	
		if (!fileName.endsWith(".txt")) {
			throw new IOException("文件后缀有误");//Exception in thread "main" java.io.IOException: 文件后缀有误
		}
		System.out.println("路径正常,读取文件!");
	}
}

异常注意事项

多个异常使用捕获该如何处理
1.多个异常分别处理

try {
			int[] num = {1,2,3};
			System.out.println(num[3]);//java.lang.ArrayIndexOutOfBoundsException: 3
		} catch (ArrayIndexOutOfBoundsException e) {
			System.out.println(e);
		}
		
		try {
			List<Integer> list = List.of(1,2,3,4);
			System.out.println(list.get(4));// java.lang.IndexOutOfBoundsException: Index 4 out-of-bounds for length 4
		} catch (IndexOutOfBoundsException e) {
			System.out.println(e);
		}

2.多个异常一次捕捉,多次处理

一般常用处理方式
注意事项:catch里面定义的异常变量,如果有子父类关系,那么子类的异常必须写在父类的上面,否则就会报错

try {
			int[] num = {1,2,3};
			System.out.println(num[3]);//java.lang.ArrayIndexOutOfBoundsException: 3
			List<Integer> list = List.of(1,2,3,4);
			System.out.println(list.get(4));// java.lang.IndexOutOfBoundsException: Index 4 out-of-bounds for length 4
		}catch (ArrayIndexOutOfBoundsException e) {
			System.out.println(e);
		}catch(IndexOutOfBoundsException e) {
			System.out.println(e);
}

3.多个异常一次捕捉一次处理

try {
			int[] num = {1,2,3};
			System.out.println(num[3]);//java.lang.ArrayIndexOutOfBoundsException: 3
			List<Integer> list = List.of(1,2,3,4);
			System.out.println(list.get(4));// java.lang.IndexOutOfBoundsException: Index 4 out-of-bounds for length 4
		} catch (Exception e) {
			System.out.println(e);
		}

运行时异常被抛出可以不处理,既不捕获也不声明抛出。默认会给虚拟机处理,终止程序
什么时候不抛出运行时异常了,再继续执行程序

异常finally 有 return 语句 ,永远返回finally中的结果,避免该情况

public class Demo10 {
	public static void main(String[] args) {
		int a =geta();
		System.out.println(a); //100
	}
	//定义一个方法,返回变量a的值
	public static int geta() {
		int a =10;
		try {
			return a;
		} catch (Exception e) {
			System.out.println(e);
		}finally {
			//一定会执行的代码
			a = 100;
			return a;
		}
	}
}

子父类异常
如果父类抛出多个异常,子类重写父类方法时,抛出和父类相同的异常或者是父类异常的子类或者不抛出异常
父类方法没有抛出异常,子类重写父类方法时也能抛出异常,此时子类产生该异常们只能捕获处理,不能声明抛出
注意:父类异常是什么样,子类异常就什么样

public class Demo11 {
	public void show01() throws NullPointerException,ClassCastException{}
	public void show02() throws IndexOutOfBoundsException{}
	public void show03() throws IndexOutOfBoundsException{}
	public void show04(){}
}
class Demo11zi extends Demo11{
		//子类重写父类方法时,抛出和父类相同的异常
		@Override
		public void show01() throws NullPointerException,ClassCastException{}
		//子类重写父类方法时,抛出父类异常的子类
		@Override
		public void show02() throws ArrayIndexOutOfBoundsException{}
		//子类重写父类方法时,不抛出异常
		@Override
		public void show03(){}
		//父类方法没有抛出异常,子类重写父类方法时也能抛出异常 ,此时子类产生该异常们只能捕获处理,不能声明抛出
		@Override
//		public void show04() throws Exception{}
		public void show04() {
			//,此时子类产生该异常们只能 捕获处理,不能声明抛出
			try {
				throw new Exception("编译期异常");
			} catch (Exception e) {
				e.printStackTrace();
			}
		}		
}

自定义异常类

Java提供的异常类不够 我们使用,需要自己定义一些异常类 (例如年龄不能为负之类的异常)

格式:
public class xxxException extends Exception 或者RuntimeException{
添加一个空参数的构造方法
添加一个带异常信息的构造方法
}

注意:
1.自定义异常类,一般都是以Exception结尾,说明该类是一个异常类
2.自定义异常类,必须继承Exception 或者 RuntimeException
继承Exception:那么自定义的异常类就是一个编译器异常,如果方法内部抛出了编译期异常,就必须处理这个异常要么throws 要么try…catch
继承 RuntimeException:那么自定义的异常类,就是一个运行期异常,无需处理,交给虚拟机处理(中断处理)

public class RegisterException extends Exception{
	//添加一个空参数的构造方法
	public RegisterException () {
		
	}
	//添加一个带异常信息的构造方法
	/*
	 *  查看源码发现,所有异常类都会有一个带异常信息的构造方法,方法内部会调用父类带异常信息的构造方法,让父类处理来处理异常信息	
	 */
	public RegisterException (String s) {
		super(s);
	}
}