代码在运行中发生的问题就是异常
java中把多种异常封装多个类,当程序出现问题时候,就会创建异常类对象并且抛出相关信息
异常体系:
Throwable类是Java中所有错误或异常的父类
Throwable的子类Error类是所有错误的父类
Throwable的另一个子类Exception类是所有异常的父类
在开发中,相对来说,我们并不关心错误,而更关心异常
异常和错误的区别(通俗解释):
异常:相当于一个人感冒了,吃药睡觉等进行相应的操作即可痊愈,不修改代码处理掉异常,程序还可以执行
错误:相当于一个人得了绝症,无法治愈,必须修改代码,程序才可以执行
所以可以这样理解:我们并不关心患了绝症的人,而会想方法去治愈得了小病的人
示例:‘
package demo; public class Demo {
public static void main(String[] args) {
function1();
function2();
}
public static void function1(){
//异常
int[] arr = {1,2};
System.out.println(arr[3]);
//输出:
//Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 3
//at demo.Demo.function1(Demo.java:11)
//at demo.Demo.main(Demo.java:5)
}
public static void function2(){
//错误
int[] arr = new int[666666666];
System.out.println(arr);
//输出:
//Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
//at demo.Demo.function2(Demo.java:19)
//at demo.Demo.main(Demo.java:6)
}
}
一旦异常被抛出了,后边的程序都不会执行,因此要想办法解决这个问题
异常中的关键字throw:
写一段代码,存在空指针异常
package demo; public class ExceptionDemo {
public static void main(String[] args) {
int [] arr = null;
function(arr);
}
public static int function(int[] arr){
int i = arr[arr.length-1];
return i;
}
}
加入throw关键字,抛出异常:
package demo; public class ExceptionDemo {
public static void main(String[] args) throws Exception {
int [] arr = null;
int i = function(arr);
System.out.println(i);
}
public static int function(int[] arr) throws Exception{
//如果方法中有异常,需要在方法上声明:throws Exception
//抛出的类型和声明的异常类应当一致
//main调用这个方法,也需要加入声明
if(arr == null){
//抛出异常,告诉调用者
//手动抛出异常
throw new Exception("传递数组不存在");
}
if(arr.length == 0){
throw new Exception("传递的数组中没有元素");
}
int i = arr[arr.length-1];
return i;
}
}
/*
输出:
Exception in thread "main" java.lang.Exception: 传递数组不存在
at demo.ExceptionDemo.function(ExceptionDemo.java:16)
at demo.ExceptionDemo.main(ExceptionDemo.java:6)
*/
这里注意,如果是运行时异常(RuntimeException),不需要throws声明 ,
而且如果发生这种异常,那么必须改代码,否则代码无意义
try catch方法异常处理:
package demo; public class ExceptionDemo {
public static void main(String[] args) {
int[] arr = null;
try {
int i = function(arr);
System.out.println(i);
} catch (NullPointerException ex) {
System.out.println(ex);
} catch (ArrayIndexOutOfBoundsException ex) {
System.out.println(ex);
}
System.out.println("结束");
} public static int function(int[] arr) throws NullPointerException, ArrayIndexOutOfBoundsException {
if (arr == null) {
// 手动抛出空指针异常
throw new NullPointerException("数组不存在");
}
if (arr.length < 6) {
// 手动抛出数组索引越界异常
throw new ArrayIndexOutOfBoundsException("数组没有6索引");
}
return arr[6];
}
}
/*
输出:
java.lang.NullPointerException: 数组不存在
结束
*/
这里发现,虽然程序有异常,但是依然执行后面的代码,这就是异常处理
catch的顺序:
平级:
package demo;/*
* catch处理的注意事项:
* 有顺序
* 当异常是平级的时候,即没有继承关系的时候,没有顺序限制
*/
public class ExceptionDemo {
public static void main(String[] args) {
try { } catch (NullPointerException e) { } catch (ArrayIndexOutOfBoundsException e) { }
}
}
有继承关系:
package demo;
//有继承关系:越高级的写在越靠后
public class ExceptionDemo {
public static void main(String[] args) {
try { } catch (NullPointerException ex) { } catch (RuntimeException ex) { } catch (Exception ex){ }
}
}
finally:
package demo; //finally:必须执行
//是否有异常都会执行
public class ExceptionDemo {
public static void main(String[] args) {
try {
function(1);
} catch (Exception ex) {
System.out.println(ex);
} finally {
System.out.println("代码必须执行");
}
} public static void function(int a) throws Exception {
if (a == 0) {
throw new Exception();
}
System.out.println(a);
}
}
//无论传入什么,都会输出:代码必须执行
注意事项:
1.父类的方法如果抛出异常,子类重写后可以不抛出异常,但是子类如果要抛出异常,这个异常的继承关系不能大于父类的异常
自定义异常:
有时候需要一些自定义的异常,这里做一个示例:
package demo; public class FuShuException extends RuntimeException {
public FuShuException(String string){
super(string);
}
public FuShuException(){}
}
package demo; //设置一个情景,计算两门成绩的平均数,不能为负数
//如果是一个负数,就抛出异常
public class ExceptionDeno {
public static void main(String[] args) throws FuShuException {
try {
int i = function(10, 97);
System.out.println(i);
} catch (FuShuException ex) {
ex.printStackTrace();
}
} public static int function(int a, int b) throws FuShuException {
if (a < 0 || b < 0) {
throw new FuShuException("成绩不为负");
}
int sum = a + b;
return sum / 2;
}
}