异常处理
下标越界
空指针
类型转换异常
数字格式化
算术异常(数学异常)
编程界:
除数为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关键字
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;
}
}