异常的父类Throwable
1、子类Error:虚拟机错误、线程死锁(出现这种错误程序彻底挂掉)
2、子类Exception:编码、环境、用户操作输入出现问题
(1)非检查异常(RuntimeException):
由java虚拟机自动抛出自动捕获
一般是代码的逻辑有问题
例如:
引用了空对象的方法(空指针异常NullPointerException)、数组越界(数组下标越界异常ArrayIndexOutOfBoundsException)、错误的类型转换(类型转换异常ClassCastException)、算数方面(算数异常ArithmeticException)等
(2)检查异常
需要程序员手动添加捕获语句
例如:
文件不存在(文件异常IOException)、链接错误(SQL错误SQLException)
3、try。。。catch。。catch。。catch。。
(1)代码中从上到下为从子类到父类(catch的顺序)
4、try。。。catch。。catch。。。finally
finally中为善后的工作,如关闭文件等
finally语句是在try catch语句中的return语句返回之前执行的
5、e.printStackTrace()打印异常
6、
(1)throws Exception写在会抛出异常的方法参数后面
(2)throw new Exception写在方法体内
(3)当方法可能抛出异常时 ,必须在调用方法出使用try。。catch语句
或者对调用函数进行throws Exception的声明,将异常抛给更上一层的调用者进行处理
7、java中的异常类
8、自定义异常:必须继承Exception类或者其子类
9、异常链实例
public class ChainTest {
/**
* test1()抛出“喝大了”异常
* test2():调用test1(),捕获“喝大了”异常,并且包装成运行时异常,继续抛出
* main方法中,调用test2(),尝试捕获test2()方法抛出的运行时异常
* @param args
*/
public static void main(String[] args) {
ChainTest cT = new ChainTest();
try{
cT.test2();
}catch(Exception e){
e.printStackTrace();
}
}
public void test1() throws DrunkException{
throw new DrunkException("喝车别开酒!");
}
public void test2(){
try {
test1();
} catch (DrunkException e) {
// RuntimeException newExc = new
// RuntimeException("司机一滴酒,亲人两行泪!");
// newExc.initCause(e);
//另一种写法
RuntimeException newExc = new RuntimeException(e);
throw newExc;
}
}
}
10、捕获到的异常可以在当前方法中的catch块中处理也可以抛出给调用这处理
11、总结
(1)处理运行时异常时,采用逻辑去合理的规避同时 辅助try catch处理
(2)在多重catch块后面,可以加一个catch(Exception)来处理可能会被遗漏的异常
(3)对于不确定的代码,也可以加上try catch ,处理潜在的异常
(4)尽量去处理异常 ,切忌只是简单的调用printStackTrace()去打印输出
(5)具体如何处理异常,要根据不同的业务需求和异常类型去决定
(6)尽量添加finally语句去释放占用的资源
12、应用实例
import java.util.Scanner;
public class BookManagerEasy {
private static Scanner console = new Scanner(System.in);
public static void main(String[] args) {
//定义”图书“数组
String[] books = { "C语言", "数据结构", "汇编语言", "高数", "大学语文", "毛概" };
while (true) {
System.out.println("输入命令:1-按照名称查找图书;2-按照序号查找图书");
String book;
try {
//取得整型命令
int command = inputCommand();
//根据不同命令值,进行不同操作
switch (command) {
case 1://按照图书名称选择图书
book = getBookByName(books);
System.out.println("book:" + book);
break;
case 2://按照图书序号(数组下标)选择图书
book = getBookByNumber(books);
System.out.println("book:" + book);
break;
case -1://返回值为-1,说明输入有误
System.out.println("命令输入错误!请根据提示输入数字命令!");
continue;
default://其他值的命令均认为是错误命令
System.out.println("命令输入错误!");
continue;
}
break;//退出程序
} catch (Exception bne) {
//捕获”图书不存在异常“时,要求重新输入命令
System.out.println(bne.getMessage());
continue;
}
}
}
//按照图书名称查询图书
private static String getBookByName(String[] books)
throws Exception {
System.out.println("输入图书名称:");
//获取输入的图书名称
String name = console.next();
for (int i = 0; i < books.length; i++) {
if (name.equals(books[i]))
//输入的名称与某一图书名称匹配,返回该图书
return books[i];
}
//若无匹配,抛出”图书不存在异常“
throw new Exception("图书不存在!");
}
//根据图书序号(数组下标)查询图书
private static String getBookByNumber(String[] books)
throws Exception {
while (true) {
System.out.println("输入图书序号:");
try {
//获取输入的图书序号(数组下标)
int index = inputCommand();
//若返回值为-1
if(index == -1){
System.out.println("命令输入错误!请根据提示输入数字命令!");
continue;
}
//若不出现”数组下标越界异常“,则返回相应位置的图书
String book = books[index];
return book;
} catch (ArrayIndexOutOfBoundsException e) {
//输入的序号不存在(引发”数组下标越界异常“),则抛出”图书不存在异常“
Exception bookNotExists = new Exception("图书不存在!");
bookNotExists.initCause(e);
throw bookNotExists;
}
}
}
//从控制台输入命令,用于输入命令和输入图书序号
private static int inputCommand(){
int command;
try {
command = console.nextInt();
return command;
} catch (Exception e) {
//若输入字符型或者字符串,则抛出异常,捕获该异常,抛出”错误命令异常“
console = new Scanner(System.in);//阻塞式方法,执行到这句会等待输入,否则不会执行下一句。
//返回-1
return -1;
}
}
}