JavaSE学习值之--认识异常(下)

简介: JavaSE学习值之--认识异常(下)

JavaSE学习值之--认识异常(下)+https://developer.aliyun.com/article/1413510

四.异常的处理

1.防御式编程

   错误的代码总是存在且不可避免地,我们要做的就是当程序出现异常的时候及时告诉程序员,来帮助他修改代码,主要有两种思路:

1.LBYL: Look Before You Leap. 在操作之前就做充分的检查. 即:事前防御型

 即对每一个流程都充分想到其可能会发生的错误,及时处理

boolean ret = false;
ret = 登陆游戏();
if (!ret) {
处理登陆游戏错误;
return;
}
ret = 开始匹配();
if (!ret) {
处理匹配错误;
return;
}
ret = 游戏确认();
if (!ret) {
处理游戏确认错误;
return;
}
ret = 选择英雄();
if (!ret) {
处理选择英雄错误;
return;
}

 每进行一步就进行一次检查,但这种方式使正确代码和异常检查的代码混在一起,使得整个代码很混乱,不推荐

2.EAFP: It's Easier to Ask Forgiveness than Permission. "事后获取原谅比事前获取许可更容易". 也就是先操 作, 遇到问题再处理. 即:事后认错型

try {
登陆游戏();
开始匹配();
游戏确认();
选择英雄();
载入游戏画面();
...
} catch (登陆游戏异常) {
处理登陆游戏异常;
} catch (开始匹配异常) {
处理开始匹配异常;
} catch (游戏确认异常) {
处理游戏确认异常;
} catch (选择英雄异常) {
处理选择英雄异常;
} catch (载入游戏画面异常) {
处理载入游戏画面异常;
}

 EAFP的思维简单来说就是管他三七二十一,先执行再说,产生的异常事后处理即可

这种思维是的操作部分和异常处理部分分离,让程序员更加模块化的思考,更有利于代码的编写!!!

在Java中,异常处理主要的5个关键字:throw、try、catch、final、throws。

 在编写程序时,如果程序中出现错误,此时就需要将错误的信息告知给调用者,比如:参数检测

 在Java中,可以借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者。具体语法如下:

throw new XXXException("异常产生的原因")

举例:

public static int[] arr ={1,2,3};
    public static int getElem(int index) {
        if(index < 0 || index > arr.length) {
            throw new ArrayIndexOutOfBoundsException(index + "索引不正确,数组越界");
        }
        return arr[index];
    }
    public static void main(String[] args) {
        int[] arr ={1,2,3};
        System.out.println(getElem(0));// 正确  输出1
        System.out.println(getElem(10));// 异常
        System.out.println(getElem(1));// 不输出
    }

注意:

  • throw必须写在方法内部
  • 抛出的异常必须是Exception或Exception的子类
  • 如果抛出的异常是RuntimeException或其子类,不需要处理,直接交给JVM处理
  • 如果抛出的异常是受查异常,必须通过throws进行处理,否则无法通过编译
  • .异常一旦抛出,其后的代码就不会执行

五.异常的捕获

  异常的捕获 就是异常处理的具体方式,包括两种方式

  • try,catch语句
  • throws声明

1.异常声明throws

 处在方法声明时参数列表之后,当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛 给方法的调用者来处理。即当前方法不处理异常,提醒方法的调用者处理异常。

语法格式: 修饰符 返回值类型 方法名(参数列表) throws 异常类型1,异常类型2...{ }

比如上文的Cloneable接口的实现:

注意:

  • throws必须跟在方法的参数列表之后
  • 声明的异常必须是 Exception 或者 Exception 的子类
  • 方法内部抛出了多个异常,throws也要抛出多个异常,异常之间通过”,“隔开;如果异常之间存在父子关系,只需声明父类异常即可
public class Config {
File file;
// public void OpenConfig(String filename) throws IOException,FileNotFoundException{
// FileNotFoundException 继承自 IOException
public void OpenConfig(String filename) throws IOException{
if(filename.endsWith(".ini")){
throw new IOException("文件不是.ini文件");
}
if(filename.equals("config.ini")){
throw new FileNotFoundException("配置文件名字不对");
}
// 打开文件
}
public void readConfig(){
}
}
  • 调用声明抛出异常的方法时,调用者必须对该异常进行处理,或者继续使用throws抛出

2 try-catch捕获并处理

 throws声明异常并没有处理异常,只是将异常交给调用者去处理,一直传递给JVM虚拟机;要想真正处理异常,就需要try-catch语句

注意:

1.try中存放可能出现异常的代码,也就是说不一定会抛出异常

2.catch()存放异常的类型,{}内部存放处理的代码,处理完成后会继续执行后续代码;如果没有捕获到异常,则catch里的语句并不会被执行

3.finally中的语句一定会被执行,

4.打印异常有三种

5.如果抛出异常类型与catch时异常类型不匹配,即异常不会被成功捕获,也就不会被处理,继续往外抛,直到 JVM收到后中断程序----异常是按照类型来捕获的

 

6.如果异常之间存在父子关系,子类异常被捕获时在前,父类在后,如果顺序颠倒,子类异常的捕获就没有任何意义了

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 (Exception e) { // Exception可以捕获到所有异常
e.printStackTrace();
}catch (NullPointerException e){ // 永远都捕获执行到
e.printStackTrace();
}
System.out.println("after try catch");
}

7.无论如何,finally中的代码一定会被执行

public static int[] arr ={1,2,3};
    public static int getElem(int index) {
        return arr[index];
    }
    public static void main(String[] args) {
        int[] arr ={1,2,3};
        try{
            System.out.print(getElem(100));// 异常
        }catch (Exception e) {
            e.printStackTrace();
        }finally {
            System.out.println("finally中的代码一定会被执行");
        }
}

 在写程序时,有些特定的代码,不论程序是否发生异常,都需要执行,比如程序中打开的资源:网络连接、数据库 连接、IO流等,在程序正常或者异常退出时,必须要对资源进进行回收。另外,因为异常会引发程序的跳转,可能 导致有些语句执行不到,finally就是用来解决这个问题的。

思考:那我们该如何记录并处理异常呢?什么时间去处理异常呢?

 在目前我们所写的代码中,常常采取记录错误日志的方式来记录异常,通过编译器的报错告诉程序员出错的地方

六.如何设计一个自定义类型的异常?

 将你可能出现的异常设计为一个继承ExceptionRunTimeException的类,添加一个带参数的构造方法(用于打印错误信息),在调用时,一旦出错就抛出对应的异常即可

 注意:继承Exception的类是受查异常,必须在调用的方法声明中添加throws该异常

继承RunTimeException的类是非受查异常,不需要添加声明

e.printStackTrace();打印异常信息

class UsernameException extends RuntimeException {
    public UsernameException(String message) {
        super(message);
    }
}
class PasswdException extends RuntimeException {
    public PasswdException(String message) {
        super(message);
    }
}
class Login {
    private String Username;
    private String Passwd;
    public static void logininfo(String Username,String Passwd) {
        if(!Username.equals("admin")) {
            throw new UsernameException("用户名错误");
        }else if(!Passwd.equals("123456")) {
            throw new PasswdException("密码错误");
        }
    }
}
public class Test1 {
    public static void main(String[] args) {
        try {
            Login.logininfo("admin2","123456");
        }catch (UsernameException ex) {
            ex.printStackTrace();
        }catch (PasswdException ex) {
            ex.printStackTrace();
        }
    }
}

异常总结:

1.throw抛出一个异常

 用于参数检测  一旦出错就可以达到终止程序运行的操作

比如传递数组和数组的下标

1.如果数组是null  直接抛出空指针异常,程序停止运行

2.如果下标超过数组的范围,直接抛出数组越界异常,程序停止运行

之前:出现这种错误,只打印错误的信息,比如arr == null  sout("数组为空"),这种方式太温柔了,只告诉你有错误,但不会终止运行;直接抛异常,既能知道错误的信息,又能终止程序的运行

2.异常处理关键字的理解

throws:是一种甩锅行为,通过在方法的生命中添加throws+异常种类,将这个可能出现的异常交给调用者处理,而调用者可以在他的方法声明时添加throws异常,交给jvm去处理,就是一种甩锅,但很方便,不需要设置try catch

try catch finally:   throws并没有处理异常,只是甩锅给其他人;try catch是一种自己解决异常的方法,通过catch捕获异常并记录异常的日志  所谓的捕获异常就是解决该异常(异常都是被抛出的,catch接受抛出的异常,给他限制在笼子里,这样就没有异常了)

finally:有些代码无论程序是否异常都必须要被执行(比如电脑就算连不上网,也要能看到界面啊?再比如,校园网欠费,无法上网,但是我还能打开缴费网站),如果打不开那我怎么解决上网的问题?所以finally就是用来解决这种任何情况下都要执行的代码(也就是说无论你欠不欠费都能打开缴费网站)

对于已受查异常来说,其异常会在编译的时候发现,必须通过tycatch语句捕获或者在方法签名使用throws抛出

3.异常处理的流程

  • 程序先执行 try 中的代码 如果 try 中的代码出现异常, 就会结束 try 中的代码,
  • 看和 catch 中的异常类型是否匹配. 如果找到匹配的异常类型, 就会执行 catch 中的代码
  • 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
  • 无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
  • 如果上层调用者也没有处理的了异常, 就继续向上传递.
  • 一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止

目录
相关文章
|
4月前
|
数据采集 人工智能 运维
训练时间可缩短50%!阿里云推出自动驾驶模型加速框架
阿里云推出PAI-TurboX框架,专为自动驾驶领域模型训练与推理加速设计,可提升感知、规划控制及世界模型效率,缩短训练时间50%以上,并已广泛应用于多家车企。
215 0
训练时间可缩短50%!阿里云推出自动驾驶模型加速框架
|
12月前
|
负载均衡 监控 应用服务中间件
除了 Nginx,还有以下一些常见的负载均衡工具
【10月更文挑战第17天】这些负载均衡工具各有特点和优势,在不同的应用场景中发挥着重要作用。选择合适的负载均衡工具需要综合考虑性能、功能、稳定性、成本等因素。
1694 56
|
SQL 关系型数据库 MySQL
PHP与MySQL交互之基础教程
【8月更文挑战第31天】 在数字世界中,数据是推动一切的核心力量。本文将引导你探索PHP与MySQL的协同工作,通过实际代码示例,展示如何建立连接、执行查询以及处理结果集。无论你是初学者还是希望巩固知识的开发者,这篇文章都将为你提供宝贵的实践知识。
|
12月前
|
NoSQL Redis
redis 的 key 过期策略是怎么实现的(经典面试题)超级通俗易懂的解释!
本文解释了Redis实现key过期策略的方式,包括定期删除和惰性删除两种机制,并提到了Redis的内存淘汰策略作为补充,以确保过期的key能够被及时删除。
235 1
|
数据采集 存储 数据可视化
Pandas高级教程:数据清洗、转换与分析
Pandas是Python的数据分析库,提供Series和DataFrame数据结构及数据分析工具,便于数据清洗、转换和分析。本教程涵盖Pandas在数据清洗(如缺失值、重复值和异常值处理)、转换(数据类型转换和重塑)和分析(如描述性统计、分组聚合和可视化)的应用。通过学习Pandas,用户能更高效地处理和理解数据,为数据分析任务打下基础。
1369 3
|
机器学习/深度学习 物联网 TensorFlow
使用Python实现深度学习模型:在嵌入式设备上的部署
【7月更文挑战第11天】 使用Python实现深度学习模型:在嵌入式设备上的部署
1324 2
|
机器学习/深度学习 计算机视觉
【保姆级教程|YOLOv8添加注意力机制】【2】在C2f结构中添加ShuffleAttention注意力机制并训练
【保姆级教程|YOLOv8添加注意力机制】【2】在C2f结构中添加ShuffleAttention注意力机制并训练
|
机器学习/深度学习 PyTorch 测试技术
PyTorch实战:图像分类任务的实现与优化
【4月更文挑战第17天】本文介绍了使用PyTorch实现图像分类任务的步骤,包括数据集准备(如使用CIFAR-10数据集)、构建简单的CNN模型、训练与优化模型以及测试模型性能。在训练过程中,使用了交叉熵损失和SGD优化器。此外,文章还讨论了提升模型性能的策略,如调整模型结构、数据增强、正则化和利用预训练模型。通过本文,读者可掌握基础的PyTorch图像分类实践。
|
自然语言处理 Java 测试技术
使用PostMan请求阿里云通义千问大模型
本文介绍如果通过postman调用阿里云通义千问API,然后介绍如果使用多语言集成,最后介绍了快速使用postman压测创建的API请求。
3240 2
|
XML Java 数据格式
xml与java对象的快速互转
做流程图的项目时,新的流程定义为xml的,需要对xml与java对象进行互转 查了一下activiti的转换xml方式,发现转换太麻烦了,需要一步步的解析xml 后面发现直接用jaxb就可以很快实现互转,而且现在这个jaxb在jdk内,不需要引入外部的解析xml的包 具体如下: 一.
2457 0