Java的常见异常及Demo
整理了一下Java常见的异常类型,并尽量在每一个附上例子,帮助清晰理解。
- ArithmeticException
算术异常类,常见出现在公式计算除0中,举例来说,
- ArrayIndexOutOfBoundsException
数组下标越界异常,通常出现在数组取值时,取数据下标在数组中并不存在导致。
- NullPointerException 空指针异常
最常见的异常类型,通常为指针指向null导致,可能出现在数组,循环,链表等一系列遍历,取值过程中。当对象为null时的任何操作,都会导致空指针异常。
- StringIndexOutOfBoundsException
字符串越界异常
- NegativeArraySizeException
建立元素个数为负数的数组异常类
- ArrayStoreException
数组类型不兼容异常,通常在不兼容值赋给数组元素时抛出,如将Integer赋值给字符串:
- ClassCastException 强制类型转换异常
- NumberFormatException
强制类型转换错误:字符串转换为数字异常
ClassNotFoundException和NoClassDefFoundError的区别
ClassNotFoundException是一个运行时异常。从类继承层次上来看,ClassNotFoundException是从Exception继承的,所以ClassNotFoundException是一个检查异常。
当应用程序运行的过程中尝试使用类加载器去加载Class文件的时候,如果没有在classpath中查找到指定的类,就会抛出ClassNotFoundException。一般情况下,当我们使用Class.forName()或者ClassLoader.loadClass以及使用ClassLoader.findSystemClass()在运行时加载类的时候,如果类没有被找到,那么就会导致JVM抛出ClassNotFoundException。
最简单的,当我们使用JDBC去连接数据库的时候,我们一般会使用Class.forName()的方式去加载JDBC的驱动,如果我们没有将驱动放到应用的classpath下,那么会导致运行时找不到类,所以运行Class.forName()会抛出ClassNotFoundException。
- EOFException 文件已结束异常 继承自IOException
通过这个API,我们可以得出以下信息:
- 这是一个IO异常的子类,名字也是END OF FILE的缩写,当然也表示流的末尾
- 它在表明一个信息,流已经到末尾了,而大部分做法是以特殊值的形式返回给我们,而不是抛异常
也就是说这个异常是被主动抛出来的,而不是底层或者编译器返回给我的,就像NullPointerException或IndexOutOfBoundsException一样。
我们先来看InputStream,这个输入流,当读到了结尾会怎么样,看看API介绍:
可以看到如果到达流的末尾,那么会返回-1,也就是说我们可以根据这个-1来判断是否到达流的末尾。
同样的我们看一下输入流的包装类BufferedReader,它有一个读一行的方法:
也可以发现当读到流的末尾,通过返回值null来告诉我们到达流的末尾了,也就是说通过返回一个不可能的值来表示到达流的末尾。
可以感觉到EOFException的用意,因为我们可以往流中放入null值,所以我们没法找到一个不可能的值来表示到达流的末尾,所以只能通过抛异常的方式来告诉用户到达末尾了。
所以说ObjectInputStream可以自己判断流是否到达末尾,但是它无法告诉我们,我们不能替代他们读取这个标记,不然ObjectInputStream将识别不了下一个内容的实际类型。
所以呢,对于这种异常的一般解决方法就是,捕获,可以记录日志,也可以不做处理,捕获异常以后,把之前读到的数据进行后续的处理就可以了,因为那就是所以的数据。还有就是如果打算记录日志,不要把它的堆栈信息打印出来,容易给人以错觉。毕竟EOFException实质上只是一个消息而已。
当然抛异常的做法还是有一些偏激,但是当ObjectInputStream在不知道读取对象数量的情况下,确实无法判断是否读完,除非你把之前写入对象流的数量记录下来。所以说出现这个异常时就认真分析一下,这个异常是不是代表一个信息。
- FileNotFoundException 找不到文件异常
- SQLException
当使用 JDBC 与数据源进行交互的时候遇见错误的时候,将会抛出名为 SQLException 的异常。一个 SQLException 的异常里面包含以下信息,用于帮助我们更好的定位错误。
- IllegalAccessException
安全权限异常,一般来说,是由于java在反射时调用了private方法所导致的。
构造参见另一篇文章:ava反射--Field用法实践与IllegalArgumentException构造
- InstantiationException
实例化异常。当试图通过newInstance()方法创建某个类的实例,而该类是一个抽象类或接口时,抛出该异常。
- NoSuchFieldException
getField(String name) 或getFields() 获取非 public 的变量,编译器会报 java.lang.NoSuchFieldException 。
set(Object obj, Object value) 时,变量访问检查导致的 IllegalAccessException。由于 Field 继承自 AccessibleObject , 我们可以使用 AccessibleObject.setAccessible() 方法告诉安全机制,这个变量可以访问即可解决,如field.setAccessible(true)。
- java.lang.StackOverflowError和java.lang.OutOfMemoryError错误
在java虚拟机规范中,虚拟机栈和本地方法栈都会出现StackOverflowError和OutofMemoryError,程序计数器是java虚拟机中唯一一块不会产生error的内存区域。
StackOverflowError代表的是,当栈深度超过虚拟机分配给线程的栈大小时就会出现此error。
OutofMemoryError代表的是,当再申请新的内存时,虚拟机分配给线程的内存大小中无法再分配新的内存,就会出现此error。
Java内存管理详细说明移步:java内存区域与内存溢出异常(内存泄漏)
在单线程操作中,无论是栈深度无限增加,还是栈帧(每个方法调用执行时都会在栈中创建一个栈帧,用来存储局部变量,操作数栈,动态链表,方法出口等信息)占的空间太大,都出现的是StackOverflowError。
不断创建新的线程会出现OutofMemoryError的错误,这个就不要run了,会把系统内存打满。