4. 异常
4.1. 异常简介
- 是什么:异常就是在程序的运行过程中出现的不正常现象导致了程序中断
- 如何表示:
- 思考:程序中出现异常应该如何描述?
- 异常名称
- 异常的描述信息
- 异常出现的位置
- 所以Java通过对共性抽取出来一个类叫做Throwable
-
- Throwable:可抛出的:既然出现错误就需要让程序员知道,也可以理解为(可报错的)
- Error:不可预期的错误
- Exception:可预期的错误
- 编译时被检查异常:Exception下的其他子类
- 运行时异常:RuntimeException及其子类
- Throwable:可抛出的:既然出现错误就需要让程序员知道,也可以理解为(可报错的)
-
- 思考:程序中出现异常应该如何描述?
4.2. 异常处理
4.2.1. 捕捉
是什么:try..catch
(尝试运行,如果运行出现问题,则在catch中进行捕捉)
怎么用:
try{
//需要被检测的代码;
}catch(异常类型 e){
//异常处理代码;
System.out.println("message:"+e.getMessage());//获取的是异常的信息。
System.out.println("toString:"+e.toString());//获取的是异常的名字+异常的信息。
e.printStackTrace();//打印异常在堆栈中信息;异常名称+异常信息+异常的位置。
}finally{
//一定会运行的代码(一般用来关闭资源);
}
注意:
- 抛出几个就处理几个,如果有父子关系,则子类异常要先catch
4.2.2. 抛出
是什么:throw
(我啥也不管,一旦出问题,直接抛给调用者)
怎么用:throw new 异常类名();
(方法中)
示例:
public class DemoThrow {
public static void main(String[] args) {
int a = DemoThrow.div(4,0);
System.out.println(a);
}
public static int div(int a,int b)
{
if(b==0)
throw new ArithmeticException("异常信息:除数不能为0");//抛出具体问题,编译时不检测
return a/b;
}
}
注意:
- 什么时候定义
try
,什么时候定义throws
呢- 功能内部如果出现异常,如果内部可以处理,就用
try
; - 如果功能内部处理不了,就必须
throws
声明出来,让调用者处理。
- 功能内部如果出现异常,如果内部可以处理,就用
- 在子父类进行覆盖时,异常如何处理
- 当子类覆盖父类的方法时,如果父类的方法抛出了异常,那么子类的方法要么不抛出异常要么抛出父类异常或者该异常的子类,不能抛出其他异常。
- 如果父类抛出了多个异常,那么子类在覆盖时只能抛出父类的异常的子集。
- 如果父类或者接口中的方法没有抛出过异常,那么子类是不可以抛出异常的,如果子类的覆盖的方法中出现了异常,只能try不能throws。
- 如果这个异常子类无法处理,已经影响了子类方法的具体运算,这时可以在子类方法中,通过throw抛出RuntimeException异常或者其子类,这样,子类的方法上是不需要
throws
声明的。
- 上面讲到了异常分为编译时被检查异常和运行时异常,他们的区别在使用上来说,如果是编译时被检查异常,则方法上需要声明
- 是什么:用于表示当前方法不处理异常,而是提醒该方法的调用者来处理异常
- 怎么用:
修饰符 返回值类型 方法名(参数) throws 异常类名1,异常类名2 ... { }
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
public class DemoThrows {
public static void main(String[] args) throws FileNotFoundException{
readFile();
}
public static void readFile() throws FileNotFoundException {
InputStream is = new FileInputStream("E:/iodemo/ch01.txt");
}
}
4.4. 自定义异常
- 是什么:除了JDK定义好的异常类外,在开发过程中根据业务的异常情况自定义异常类。
- 怎么用:
- 定义一个子类继承Exception或RuntimeException,让该类具备可抛性。
- 通过throw 或者throws进行操作。
class WrongInputException extends Exception { // 自定义的类
WrongInputException(String s) {
super(s);
}
}
class Input {
void method() throws WrongInputException {
throw new WrongInputException("Wrong input"); // 抛出自定义的类
}
}
class TestInput {
public static void main(String[] args){
try {
new Input().method();
}
catch(WrongInputException wie) {
System.out.println(wie.getMessage());
}
}
}
4.5. 常见异常
- java.lang.NullpointerException(空指针异常)
- 这个异常经常遇到,异常的原因是程序中有空指针,即程序中调用了未经初始化的对象或者是不存在的对象。经常出现在创建对象,调用数组这些代码中,比如对象未经初始化,或者图片创建时的路径错误等等。对数组代码中出现空指针,是把数组的初始化和数组元素的初始化搞混淆了。数组的初始化是对数组分配空间,而数组元素的初始化,是给数组中的元素赋初始值
- java.lang.NumberFormatException(数字转换异常)
- 当试图将一个String类型数据转换为指定的数字类型,但该字符串不满足数值型数据的要求时,就抛出这个异常。例如将String类型的数据"123456"转换为数值型数据时,是可以转换的的。但是如果String类型的数据中包含了非数字型的字符,如
123a56
,此时转换为数值型时就会出现异常。系统就会捕捉到这个异常,并进行处理
- 当试图将一个String类型数据转换为指定的数字类型,但该字符串不满足数值型数据的要求时,就抛出这个异常。例如将String类型的数据"123456"转换为数值型数据时,是可以转换的的。但是如果String类型的数据中包含了非数字型的字符,如
- java.lang.ClassNotFoundExceptio(指定的类不存在)
- 是因为类的名称和路径不正确,通常都是程序试图通过字符串来加载某个类的时候可能会引发异常。例如:调用Class.forName()、或者调用ClassLoad的finaSystemClass()、或者是LoadClass()时出现异常
- java.lang.IndexOutOfBoundsException(数组下标越界异常)
- 查看程序中调用的数组或者字符串的下标值是不是超出了数组的范围,一般来说,显示调用数组不太容易出这样的错,但隐式调用就有可能出错了,还有一种情况,是程序中定义的数组的长度是通过某些特定方法决定的,不是事先声明的,这个时候可以先查看一下数组的length,以免出现这个异常
- java.lang.IllegalArgumentException(方法的参数错误)
- 例如g.setColor(int red,int green,int blue)这个方法中的三个值,如果有超过255的会出现这个异常,如果程序中存在这个异常,就要去检查一下方法调用中的参数传递或参数值是不是有错
- java.lang.IllegalAccessException(没有访问权限)
- 当程序要调用一个类,但当前的方法即没有对该类的访问权限便会出现这个异常。如果程序中用了Package的情况下有可能出现这个异常
- java.lang.ArithmeticException(数学运算异常)
- 当数学运算中出现了除以零这样的运算就会出这样的异常。
- java.lang.ClassCastException(数据类型转换异常)
- 当试图将对某个对象强制执行向下转换,但该对象又不可转换或又不可转换为其子类的实例时将出现该异常
- java.lang.FileNotFoundException(文件未找到异常)
- 当程序打开一个不存在的文件来进行读写时将会引发该异常。该异常由FileInputStream,FileOutputStream,RandomAccessFile的构造器声明抛出,即使被操作的文件存在,但是由于某些原因不可访问,比如打开一个只有只读权限的文件并向其中写入数据,以上构造方法依然会引发异常
- java.lang.ArrayStoreException(数组存储异常)
- 当试图将类型为不兼容类型的对象存入一个Object[]数组时将引发异常
- java.lang.NoSuchMethodException(方法不存在异常)
- 当程序试图通过反射来创建对象,访问(修改或读取)某个方法,但是该方法不存在就会引发异常。
- java.lang.EOFException(文件已结束异常)
- 当程序在输入的过程中遇到文件或流的结尾时,引发异常。因此该异常用于检查是否达到文件或流的结尾
- java.lang.InstantiationException(实例化异常)
- 当试图通过Class的newInstance()方法创建某个类的实例,但程序无法通过该构造器来创建该对象时引发。Class对象表示一个抽象类,接口,数组类,基本类型 。该Class表示的类没有对应的构造器。
- java.lang.InterruptedException(被中止异常)
- 当某个线程处于长时间的等待、休眠或其他暂停状态,而此时其他的线程通过Thread的interrupt方法终止该线程时抛出该异常。
- java.lang.CloneNotSupportedException (不支持克隆异常)
- 当没有实现Cloneable接口或者不支持克隆方法时,调用其clone()方法则抛出该异常
- java.lang.OutOfMemoryException (内存不足错误)
- 当可用内存不足以让Java虚拟机分配给一个对象时抛出该错误
- java.lang.NoClassDefFoundException (未找到类定义错误)
- 当Java虚拟机或者类装载器试图实例化某个类,而找不到该类的定义时抛出该错误