什么是异常
异常指程序运行中出现的不期而至的各种状况,如:文件找不到,网络连接失败,非法参数等
异常发生在程序运行期间,他影响了正常的程序执行流程。
举例:
//无限循环的代码,在写代码时不会报错,在编译的时候会报错
public static void main(String[] args) {
new demo1().a();
}
public void a(){
b();
}
public void b(){
a();
}
}
运行结果为:
异常分类:
-
检查性异常:最具代表性的检查性异常就是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单的忽略
-
运行时异常:运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可在编译时被忽略
-
错误:错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,他们在编译时检查不到
JAVA把异常当做对象来处理,并定义一个基类Java.lang.Throwable作为所有异常的超类
在JAVA API中已经定义了许多异常类,这些异常类分为两大类,错误Error和异常Exception
Exception:
在Exception分支中有一个重要的子类RuntimeException(运行异常)
ArrayIndexOutOfBoundsException(数组下标越界)
NullPointerException(空指针异常)
ArithmeticException(算数异常)
MissingResourceException(丢失资源)
ClassNotfoundException(找不到类)等异常
Error:
Error类对象由 Java虚拟机生成并抛出,大多数错误与代码编写者所执行的操作无关,这些异常发生时,Java虚拟机(JVM)一般会选择线程终止;还有发生在虚拟机试图执行应用时,如类定义错误(NoClassDefFoundError)、链接错误(LinkageError)。这些错误是不可查的,因为它们在应用程序的控制和处理能力之外,而且绝大多数是程序运行时不允许出现的状况。
区别:
Error通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时,JAVA虚拟机(JVM)一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的处理这些异常。
捕获和抛出异常
举例:
//try,catch,finally三个同时出现,finally可以不要。
public static void main(String[] args) {
int a=1;
int b=0;
try{//监控区域
System.out.println(a/b);
}catch (ArithmeticException e){//想要捕捉的异常
System.out.println("程序出现异常,变量b不能为0");
}finally {//处理善后工作
System.out.println("finally");
}
}
运行结果:
catch()可以同时捕获多个异常,层层递进,最大的异常写在最下面,从上向下依次去执行。
举例:
public static void main(String[] args) {
int a=1;
int b=0;
try{//监控区域
System.out.println(a/b);
}catch (Error e){//捕捉异常
System.out.println("Error");
}catch (Exception c){
System.out.println("Exception");
}catch (Throwable t){
System.out.println("Throwable");
} finally {//处理善后工作
System.out.println("finally");
}
输出结果为:Exception finally
除数不能为零是ArithmeticException异常,所以代码执行第二个catch
快捷键:
自动包裹代码:可以选择多种包裹方式,选中这行代码。Ctrl+Alt+t
try {
System.out.println(a/b);
} catch (Exception e) {
e.printStackTrace();//打印错误的栈信息
} finally {
}
主动抛出异常关键字:throw throws,一般用在方法里
try {
if(b==0){
throw new ArithmeticException();
}
public static void main(String[] args) {
new test2().test(1,0);
}
public void test(int a,int b){
if(b==0){
throw new ArithmeticException();
}
System.out.println(a/b);
}
}
运行结果:主动抛出异常,程序就不再运行了
throws:
public void test(int a,int b)throws ArithmeticException{
if(b==0){
throw new ArithmeticException();
}
System.out.println(a/b);
自定异常
使用JAVA内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户还可以自定义异常。用户自定义异常类,只需要继承Exception类即可。一般用的不多。
在程序中使用自定义异常类,可分为以下几个步骤:
1.创建自定义异常
2.在方法中通过throw关键字抛出异常
3.如果在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throw关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
4.在出现异常方法的调用者中捕获并处理异常
举例:
//自定义异常类继承Exception
public class MyException extends Exception {
//传递数字>10
private int detail;
//异常的打印信息
@Override
public String toString() {
return "MyException{" +
"detail=" + detail +
'}';
}
public MyException(int a) {
this.detail=a;
}
//测试类
public class Test3 {
//可能会存在异常的方法
static void test(int a) {
if(a>10){
throw new MyException(a);//这会出现报错,提示方法try,catch和抛出
}
static void test(int a) throws MyException {
if(a>10){
throw new MyException(a);
第二种方法:
if(a>10){
try {
throw new MyException(a);
} catch (MyException e) {
e.printStackTrace();
}
主类:
public static void main(String[] args) {
try {
test(1);
} catch (MyException e) {
System.out.println("MyException=>"+e);
}
}
当输入为1时,输出结果为:
传递的数字为:1
OK
当输入为11时,输出结果为:
传递的数字为:11
MyException=>MyException{detail=11}
总结:
- 处理运行异常时,采用逻辑去合理规避同时辅助try—catch处理
- 在多重catch块后面,可以加一个最大的catch(Exception)来处理可能会遗漏的异常
- 对于不确定的代码,也可以加上try-catch,处理潜在异常
- 尽量去处理异常,切记只简单的调用printStackTrace()去打印输出
- 具体如何处理异常,根据不同的业务需求和异常类型去决定
- 尽量添加finally语句块去释放占用的资源,IO流用的较多