欢迎关注个人主页:逸狼
创造不易,可以点点赞吗~
如有错误,欢迎指出~
目录
认识异常
异常分类
举例
栈溢出错误
空指针异常(运行时异常)
编译时异常
处理异常
抛出 异常
程序本身触发异常
手动抛出异常
举例
利用try catch处理异常
多个异常捕获
finally
异常处理流程总结
自定义异常
举例
自定义UserNameException异常
自定义PassWordException异常
异常处理
认识异常
在Java中,将程序执行过程中发生的不正常行为称为异常。
Throwable:是异常体系的顶层类,其派生出两个重要的子类, Error 和 Exception
Error:指的是Java虚拟机无法解决的严重问题,比如:JVM的内部错误、资源耗尽等,典型代表: StackOverflowError和OutOfMemoryError,一旦发生回力乏术。
Exception:异常产生后程序员可以通过代码进行处理,使程序继续执行。比如:感冒、发烧。我们平时所说 的异常就是Exception。
异常分类
异常又分为运行时异常(非受查异常)和编译时异常(受查异常)。
继承RuntimeException的为运行时异常(上图中浅蓝色部),剩下的为编译时异常
注意:语法错误不属于异常。
举例
栈溢出错误
public static void fun(){
fun();
}
public static void main(String[] args) {
fun();
}
空指针异常(运行时异常)
public static void main(String[] args) {
int[]array=null;
System.out.println(array.length);
}
编译时异常
class Person implements Cloneable{
@Override//重写克隆方法
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
public static void main(String[] args) throws CloneNotSupportedException {
Person person =new Person();
Person person1=(Person)person.clone();
}
}
处理异常
在Java中,异常处理主要的5个关键字:throw、try、catch、final、throws。
抛出 异常
程序本身触发异常
System.out.println(10/0);
手动抛出异常
java借助throw手动抛出异常
throw new XXXException("异常产生的原因");
举例
public static void func(int[] array){
if(array==null){
throw new RuntimeException("传个参数看看。。。"+array);
}
}
public static void main(String[] args) {
int[] array=null;
func(array);
}
注意:
- throw必须写在方法体内部
- 抛出的对象必须是Exception 或者 Exception 的子类对象
- 如果抛出的是 RunTimeException 或者 RunTimeException 的子类,则可以不用处理,直接交给JVM来处理
- 如果抛出的是编译时异常,用户必须处理,否则无法通过编译
- 异常一旦抛出,其后的代码就不会执行。
抛出异常,但没有处理,此时这个异常最终交给JVM处理了(程序直接崩溃)
如
利用try catch处理异常
try中存放可能出现的异常代码,catch中处理异常
public static void func(int[] array) throws Exception {
if(array==null){
throw new Exception("传个参数看看。。。"+array);
}
}
public static void main(String[] args) {
try{
//存放可能出现异常的代码
int[] array=null;
func(array);
System.out.println("try中的代码不会执行");
}catch(Exception e){
System.out.println("捕获到了Exception异常!,此时可以开始处理这个异常了");
e.printStackTrace();//用于定位异常的位置
}
System.out.println("异常处理完,程序继续执行");
}
注意:
- try中的代码不会执行
- catch中的代码及 后面的代码会被执行
- 如果没有捕捉到对应的异常,就会交给JAM处理(程序直接中止)
结果
关于异常的处理方式 异常的种类有很多, 我们要根据不同的业务场景来决定.
- 对于比较严重的问题(例如和算钱相关的场景), 应该让程序直接崩溃, 防止造成更严重的后果
- 对于不太严重的问题(大多数场景), 可以记录错误日志, 并通过监控报警程序及时通知程序猿
- 对于可能会恢复的问题(和网络相关的场景), 可以尝试进行重试.
在我们当前的代码中采取的是经过简化的第二种方式. 我们记录的错误日志是出现异常的方法调用信息, 能很 快速的让我们找到出现异常的位置. 以后在实际工作中我们会采取更完备的方式来记录异常信息.
多个异常捕获
try中可能会抛出多个不同的异常对象,则必须用多个catch来捕获----即多种异常,多次捕获
但catch最终只会捕获一个异常,若还是没有捕捉到对应的异常,就会交给JAM处理
public static void main(String[] args) {
int[] arr = {1, 2, 3};
try {
System.out.println("before");
// arr = null;
System.out.println(arr[100]);
System.out.println("after");
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println("这是个数组下标越界异常");
e.printStackTrace();
} catch (NullPointerException e) {
System.out.println("这是个空指针异常");
e.printStackTrace();
}
System.out.println("after try catch");
}
也可写成这样
catch (ArrayIndexOutOfBoundsException | NullPointerException e) {
...
}
如果捕捉的异常具有父子类关系,一定是子类在前,父类在后
public static void main(String[] args) {
try{
System.out.println(10/0);
}catch(ArithmeticException e){
System.out.println("捕获到一个算数异常,可以开始处理了");
e.printStackTrace();
}catch(Exception e){
System.out.println("相当于保底的");
}
finally
不够是否抛出异常,finally一定会被执行(尽量避免在fianlly中使用return),finally一般用来释放资源。
public static void main(String[] args) {
Scanner scanner=new Scanner(System.in);
try{
int a =scanner.nextInt();
System.out.println(a/0);
}catch(ArithmeticException e){
e.printStackTrace();
System.out.println("处理异常。。。");
}
finally {
System.out.println("finally 执行了。。。。");
scanner.close();
}
}
也可以在try中实例化scanner,这样finally中就不用关闭资源了
try (Scanner scanner = new Scanner(System.in)) {
...
finally {
System.out.println("finally 执行了。。。。");
}
异常处理流程总结
- 程序先执行 try 中的代码
- 如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
- 如果找到匹配的异常类型, 就会执行 catch 中的代码
- 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
- 无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
- 如果上层调用者也没有处理的了异常, 就继续向上传递.
- 一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止.
自定义异常
java 中虽然已经内置了丰富的异常类, 但是并不能完全表示实际开发中所遇到的一些异常,此时就需要维护符合我们实际情况的异常结构.
自定义异常通常会继承自 Exception(编译时异常) 或者 RuntimeException(运行时异常)
举例
我们实现一个用户登陆功能.
自定义UserNameException异常
public class UserNameException extends RuntimeException{//需要继承一个异常类型(这里继承的时运行时异常)
//提供两种构造方法,一种是带参数的,另一种是不带参数的
public UserNameException(){
}
public UserNameException(String msg){
super(msg);
}
}
自定义PassWordException异常
public class PassWordException extends RuntimeException{
public PassWordException(){
}
public PassWordException(String s){
super(s);
}
}
异常处理
public class LogIn {
private String userName="admin";
private String password="123456";
public void loginInfo(String userName,String password)
throws UserNameException,PassWordException{//声明两个异常
if(!this.userName.equals(userName)){
throw new UserNameException("用户名异常!");
}
if(!this.password.equals(password)){
throw new PassWordException("密码异常!");
}
System.out.println("登入成功");
}
public static void main(String[] args) {
LogIn logIn=new LogIn();
try{
logIn.loginInfo("admin","123456");
}catch(UserNameException e){
System.out.println("捕捉到了UserNameException。。。");
e.printStackTrace();
}catch(PassWordException e){
System.out.println("捕捉到了PassWordException。。。");
e.printStackTrace();
}finally {
System.out.println("finally...");
}
}
}