下标越界
空指针
类型转换异常
数字格式化
算术异常(数学异常)
编程界:
除数为0
IO流,没有关闭
停电
当一个程序抛出异常时,抛异常后面的语句不再执行,类似于return的功能,终止方法的执行。
public class Ch01 {
public static void main(String[] args) { int num1 = 10; int num2 = 0; if(num2 != 0){ System.out.println(num1 / num2); }
// double num2 = 0;
// BigDecimal bigDecimal1 = new BigDecimal(13.0);
// BigDecimal bigDecimal2 = new BigDecimal(0);
// System.out.println(bigDecimal1.divide(bigDecimal2));
System.out.println("我很重要...");
}
}
最顶级的Throwable:错误,异常
Error
正常情况下,不太可能出现的。绝大多数Error都会导致程序处于非正常的状态下,很难恢复。外力的作用下,不考虑。Error是Throwable的子类。它是在Java程序处理范围之外的。
Exception
Java语言中,将程序执行中发生的不正常的情况称之为异常。
编译期异常:写代码的时候,抛异常。如果编译器不解决,会编译不通过,一直报红。
运行期异常:RuntimeException,运行时会抛异常,平时没事
自定义异常:
Java中异常机制,但是结合实际业务。
怎么自定义异常?
所有异常必须是Throwable的子类(大材小用,没必要)
如果要定义一个编译期异常,需要继承Exception类。
如果要定义一个运行期异常,需要继承RuntimeException类。
public class Ch02 {
public static void main(String[] args) {
int num1 = 10;
int num2 = 2;
try{
// 把有可能抛异常的代码放到try语句块里
System.out.println(num1 / num2);
}catch (Exception e){
System.out.println("除数不能为0");
e.printStackTrace();
}
System.out.println("我很重要...");
}
}
try...catch
在一个语句块中,如果使用throw抛出一个编译期异常,就必须在方法的声明处使用throws关键字来标记异常类型
还有一种处理方式,就是直接try...catch
我们为什么要手动抛异常?
因为要配合全局异常处理机制来解决问题
public class Ch03 {
public static void fun(int i,int j) throws MyException { if(j == 0){ throw new MyException("除数不能为0"); } System.out.println(i / j); } public static void main(String[] args) { try { fun(1,1); } catch (MyException e) { // 打印异常信息 e.printStackTrace(); } }
}
throw语句是可以当做方法的返回值的。
在一个有返回值的方法中,如果有条件分支,一定要保证每种情况下都有返回值,哪怕是抛异常。
开发中,大部分情况下使用的都是运行期异常!!!
public class Ch04 {
public static String show(String str) { if(!Objects.isNull(str)){ return str.concat("hello"); }
// throw new RuntimeException("参数不能是空");
throw new ServiceException(101,"账号不能为空。");
}
public static void main(String[] args) { show(null); }
}
一个异常被抛出去之后会继续被调用这个方法的方法捕获或抛出,异常会扩散。
只要说解决异常,处理异常,捕获,就是try...catch。
class A{
public void a() {
throw new ServiceException(201,"业务异常...");
}
}
class B {
public void b() {
A aa = new A();
try {
aa.a();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("b方法的语句");
}
}
class C {
public void c(){
B b = new B();
b.b();
System.out.println("c方法的语句..");
}
}
public class Ch05 {
public static void main(String[] args) { C c = new C(); c.c(); System.out.println("Ch05的语句"); }
}
throws
如果一个方法没有捕获一个编译期异常,该方法必须使用throws来声明。
throws并不是真正的解决了异常,只是把异常抛给了下一级调用者。
throws出现在什么地方?
方法的声明处,抛出多个异常,用逗号隔开。
public class Ch06 {
public void show() throws MyException,RuntimeException,NullPointerException,IndexOutOfBoundsException { } public void info() throws MyException { show(); } // JVM public static void main(String[] args) { try { new Ch06().info(); } catch (MyException e) { throw new RuntimeException(e); } }
}
finally用来创建在try代码块后面执行的代码块
无论是否发生异常,finally代码块中的代码一定会执行。一般finally中的代码都是用来释放资源。
try...catch...finally
public class Ch07 {
public static void main(String[] args) {
int num1 = 10;
int num2 = 2;
String str = null;
try {
System.out.println(num1 / num2);
System.out.println(str.length());
// main(args);
}finally{
System.out.println("finally...");
}
}
}
catch
catch可以写多个,有顺序问题。
先写小的,再写大的
public class Ch08 {
public static void main(String[] args) {
int num1 = 10;
int num2 = 0;
try {
System.out.println(num1 / num2);
// 开发角度来说,应该写指定的。
} catch (ArithmeticException e) {
// e.printStackTrace();
System.out.println("除数不能为0");
} catch (Exception e){
System.out.println("未知错误");
}
}
}
方法的重写
重写的方法不能抛出比被重写方法更大的异常类型
interface Inter01 {
void show() throws Exception;
}
public class Ch09 implements Inter01 {
@Override public void show() throws NullPointerException { }
}
编译期异常
public class MyException extends Exception {
public MyException(String message) { super(message); }
}
运行期异常
public class MyRuntimeException extends RuntimeException {
public MyRuntimeException(String message) { super(message); }
}
try...catch..finally执行顺序
finally永远是在最后执行的
public class Question01 {
public static int test1(){ int i = 10; try { i++; System.out.println("try:" + i); }catch (Exception e){ i--; System.out.println("catch:" + i); }finally { i = 100; System.out.println("finally:" + i); } return i; } public static int test2(){ int i = 10; try { i++;
// System.out.println("try:" + i);
throw new Exception();
}catch (Exception e){
i--;
System.out.println("catch:" + i);
}finally {
i = 100;
System.out.println("finally:" + i);
}
return i;
}
// try---finally---return
public static int test3(){
int i = 10;
try {
i++;
System.out.println("try:" + i);
return i;
}catch (Exception e){
i--;
System.out.println("catch:" + i);
return i;
}finally {
i = 100;
System.out.println("finally:" + i);
}
}
// try...finally...return
public static int test4(){
int i = 10;
try {
i++;
System.out.println("try:" + i);
// System.exit(-1);
return i;
// i = 100;
// System.out.println("finally:" + i);
}catch (Exception e){
i--;
System.out.println("catch:" + i);
return i;
}finally {
i = 100;
System.out.println("finally:" + i);
// return i;
}
}
public static void main(String[] args) {
// System.out.println(test1());
// System.out.println(test2());
// System.out.println(test3());
System.out.println(test4());
}
}
自定义异常类
错误码通常使我们自己定义的规则:
例如:
100-成功
101-账号不存在
102-账号格式不对
103-账号已存在
public class ServiceException extends RuntimeException {
// 错误码 private Integer code; // 异常信息 private String message; public ServiceException() { } public ServiceException(Integer code, String message) { this.code = code; this.message = message; } public Integer getCode() { return code; } public void setCode(Integer code) { this.code = code; } @Override public String getMessage() { return message; } public void setMessage(String message) { this.message = message; }
}