异常与错误:程序运行过程中,可能发生一些不被期望的效果,肯定会阻止我们的程序按照指令去执行这种不被预期出现的效果。
异常发生的原因有很多,通常包含以下几大类:
- 用户输入了非法数据。
- 要打开的文件不存在。
- 网络通信时连接中断,或者JVM内存溢出。
这些异常有的是因为用户错误引起,有的是程序错误引起的,还有其它一些是因为物理错误引起的。-
要理解Java异常处理是如何工作的,你需要掌握以下三种类型的异常:
- **检查性异常:**最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
- 运行时异常: 运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
- 错误: 错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。
Exception(异常)
异常的简介
所有的异常类是从 java.lang.Exception 类继承的子类。
异常类有两个主要的子类:IOException 类和 RuntimeException 类。
运行时异常(非检查性异常)
运行时异常都是RuntimeException
类及其子类异常,如NullPointerException
(空指针异常)、IndexOutOfBoundsException
(下标越界异常)等,这些异常是***不检查异常***,程序中可以选择捕获处理,也可以不处理。这些异常一般是***由程序逻辑错误引起***的,程序应该从逻辑角度尽可能避免这类异常的发生。
运行时异常的特点是Java编译器不会检查它(javac编译的时候,不会提示和发现),也就是说,当程序中可能出现这类异常,即使没有用try-catch
语句捕获它,也没有用throws
子句声明抛出它,也会编译通过。
异常 | 描述 | 示例 |
---|---|---|
InputMisMatchException | 输入类型不匹配 | int value = input.nextInt(); //输入abc |
NumberFormatException | 数字格式化异常 | int value = Integer.paseInt(“123.45”); |
NegativeArraySizeException | 数组长度为负数 | int[] array = new int[-3]; |
ArrayIndexOutOfBoundsException | 数组越界异常 | int[] array = {1,2,3}; array[5]; |
NullPointerException | 空指针异常 | Person p = null; p.study(); |
ArithmeticException | 算数异常 | int i = 10/0; |
ClassCastException | 造型异常 | Person p =new Teacher(); Student s = (Student)p; |
StringIndexOutOfBoundsException | 字符串越界 | String str = “abc”; str.charAt(5); |
IndexOutOfBoundsException | 集合越界异常 | ArrayList list = new ArrayList(); list.add(1); list.add(2); list.add(3); list.get(5); |
IllegalArgumentException | 非法参数异常 | ArrayList list = new ArrayList(-1); |
编译时异常 (检查性异常)
编译时异常是RuntimeException
以外的异常,类型上都属于Exception
类及其子类。从程序语法角度讲是必须进行处理的异常,如果***不处理,程序就不能编译通过***。如IOException
、SQLException
等以及用户自定义的Exception异常,一般情况下不自定义检查异常。
异常 | 描述 |
---|---|
ClassNotFoundException | 应用程序试图加载类时,找不到相应的类,抛出该异常。 |
CloneNotSupportedException | 当调用 Object 类中的 clone 方法克隆对象,但该对象的类无法实现 Cloneable 接口时,抛出该异常。 |
InterruptedException | 一个线程被另一个线程中断,抛出该异常。 |
异常中的方法
方法 | 说明 |
---|---|
public String getMessage() | 返回关于发生的异常的详细信息,这个消息在Throwable类的构造方法中初始化了。 |
public Throwable getCause() | 返回一个Throwable 对象代表异常原因。 |
public String toString() | 使用getMessage()的结果返回类的串级名字。 |
public void printStackTrace() | 打印toString()结果和栈层次到System.err,即错误输出流。 |
捕获异常
try{}catch(){}
使用 try 和 catch 关键字可以捕获异常。try/catch 代码块放在异常可能发生的地方。
try/catch代码块中的代码称为保护代码,使用 try/catch 的语法如下:
try{
//程序代码
}catch(Exception e){
//catch块
}
Catch 语句包含要捕获异常类型的声明。当保护代码块中发生一个异常时,try 后面的 catch 块就会被检查。
如果发生的异常包含在 catch 块中,异常会被传递到该 catch 块,这和传递一个参数到方法是一样。
public class ExcepTest{
public static void main(String args[]){
try{
int a[] = new int[2];
System.out.println("Access element three :" + a[3]);
}catch(ArrayIndexOutOfBoundsException e){
System.out.println("Exception thrown :" + e);
}
System.out.println("Out of the block");
}
}
1.try不能单独的出现
2.后面必须天剑catch或finally
3.catch有一组括号(Exception e)目的是为了捕获某一种异常
4.catch可以有很多个
捕获的异常之间没有任何的继承关系
捕获的异常需要从小到大进行捕获
5.finally不是必须存在的
finally 关键字用来创建在 try 代码块后面执行的代码块。
无论是否发生异常,finally 代码块中的代码总会被执行。
在 finally 代码块中,可以运行清理类型等收尾善后性质的语句。
throws关键字
如果一个方法没有捕获到一个检查性异常,那么该方法必须使用 throws 关键字来声明。throws 关键字放在方法签名的尾部。
public class className
{
public void deposit(double amount) throws RemoteException
{
// Method implementation
throw new RemoteException();
}
//Remainder of class definition
}
自定义异常
在 Java 中你可以自定义异常。编写自己的异常类时需要记住下面的几点。
- 所有异常都必须是 Throwable 的子类。
- 如果希望写一个检查性异常类,则需要继承 Exception 类。
- 如果你想写一个运行时异常类,那么需要继承 RuntimeException 类。
public class MyException extends Exception{
public MyException(){
}
public MyException(String msg){
super(msg);
}
}