【JavaSE】异常

news2024/11/23 22:10:07

欢迎关注个人主页:逸狼


创造不易,可以点点赞吗~

如有错误,欢迎指出~



目录

认识异常

异常分类

举例

栈溢出错误

空指针异常(运行时异常)

编译时异常

处理异常

抛出 异常

程序本身触发异常

手动抛出异常

举例

利用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 执行了。。。。");
        }

异常处理流程总结

  1. 程序先执行 try 中的代码
  2. 如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
  3. 如果找到匹配的异常类型, 就会执行 catch 中的代码
  4. 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
  5. 无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
  6. 如果上层调用者也没有处理的了异常, 就继续向上传递.
  7. 一直到 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...");
        }
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1611773.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++修炼之路之多态--多态的条件与例外,重载+重写+重定义

目录 前言 一:构成多态的条件及一些特殊情况(前提是构成父子类) 1.多态是在不同的继承关系的类对象,去调用同一函数,产生了不同的结果 2.两个条件 3.三同的两个例外 1.协变---返回值类型可以不同,但必…

【简单讲解下Stylus入门使用方法】

🎥博主:程序员不想YY啊 💫CSDN优质创作者,CSDN实力新星,CSDN博客专家 🤗点赞🎈收藏⭐再看💫养成习惯 ✨希望本文对您有所裨益,如有不足之处,欢迎在评论区提出…

从OWASP API Security TOP 10谈API安全

1.前言 应用程序编程接口(API)是当今应用驱动世界创新的一个基本元素。从银行、零售、运输到物联网、 自动驾驶汽车、智慧城市,API 是现代移动、SaaS 和 web 应用程序的重要组成部分,可以在面向客 户、面向合作伙伴和内部的应用程…

数据结构 -- 二叉树二叉搜索树

二叉树 二叉树是这么一种树状结构:每个节点最多有两个孩子,左孩子和右孩子 重要的二叉树结构 完全二叉树(complete binary tree)是一种二叉树结构,除最后一层以外,每一层都必须填满,填充时要遵…

【Windows游戏】大头菜小子抢银行游戏介绍

游戏介绍 今天写的是一款游戏,叫《大头菜小子抢银行》。 《大头菜小子抢银行》是《大头菜小子避税历险记》的续作,玩家需要联手泡菜团队,进行史上最奇怪的劫案,绑架人质,偷取珍贵财报,探索植物银行的黑暗…

MapReduce工作流程(Hadoop3.x)

MapReduce 是一种用于并行处理大规模数据集的——编程模型和处理框架。它通常用于分布式计算环境中,如Apache Hadoop。 工作流程 1. 切分阶段(Splitting): 数据集被分成多个数据块,每个数据块的大小通常在64MB到12…

nginx反向代理及负载均衡

node1192.168.136.55Nginx主负载均衡器node3192.168.136.57Web01服务器node4192.168.136.58Web02服务器node5192.168.135.131客户端(测试) nginx反向代理 1. 安装nginx 三台机器都安装nginx yum install nginx -y 2. 配置用于测试的Web服务(以下操作…

C++进阶:搜索树

目录 1. 二叉搜索树1.1 二叉搜索树的结构1.2 二叉搜索树的接口及其优点与不足1.3 二叉搜索树自实现1.3.1 二叉树结点结构1.3.2 查找1.3.3 插入1.3.4 删除1.3.5 中序遍历 2. 二叉树进阶相关练习2.1 根据二叉树创建字符串2.2 二叉树的层序遍历I2.3 二叉树层序遍历II2.4 二叉树最近…

ChatGPT研究论文提示词集合1-【主题选择与问题研究、文献综述】

点击下方▼▼▼▼链接直达AIPaperPass ! AIPaperPass - AI论文写作指导平台 目录 1.主题选择与问题定义 2.文献综述 3.书籍介绍 AIPaperPass智能论文写作平台 近期小编按照学术论文的流程,精心准备一套学术研究各个流程的提示词集合。总共14个步骤…

AI翻译英语PDF文档的3种方法

短的文章,直接丢进kimichat、ChatGPT里面很快就可以翻译完成,而且效果很佳。但是,很长的PDF文档整篇需要翻译,怎么办呢? ●腾讯交互翻译TranSmart https://transmart.qq.com/ 软件下载后,点击左边的文件翻…

天才简史——Sylvain Calinon

一、研究方向 learning from demonstration(LfD)领域的专家,机器人红宝书(Springer handbook of robotics)Robot programming by demonstration章节的合作者。主要研究兴趣包括: 机器人学习、最优控制、几…

从零自制docker-11-【pivotRoot切换实现文件系统隔离】

文章目录 busyboxdocker run -d busybox topcontainerId(docker ps --filter "ancestorbusybox:latest"|grep -v IMAGE|awk {print $1})docker export -o busybox.tar $containerId or sudo docker export 09bbf421d93f > ./busybox.tar tar -xvf busybox.tar -C …

Python | Leetcode Python题解之第41题缺失的第一个正数

题目&#xff1a; 题解&#xff1a; class Solution:def firstMissingPositive(self, nums: List[int]) -> int:n len(nums)for i in range(n):while 1 < nums[i] < n and nums[nums[i] - 1] ! nums[i]:nums[nums[i] - 1], nums[i] nums[i], nums[nums[i] - 1]for …

域名信息查询同款WHOIS源码

域名查询一般是指查询域名的whois注册信息&#xff0c;域名WHOIS是当前域名系统中不可或缺的一项信息服务。在使用域名进行Internet冲浪时&#xff0c;很多用户希望进一步了解域名、名字服务器详细信息&#xff0c;这就会用到WHOIS。 域名信息查询同款WHOIS源码

在Postgres中,如何有效地管理大型数据库的大小和增长

文章目录 一、定期清理和维护1. VACUUM和ANALYZE2. 删除旧数据和归档 二、分区表三、压缩数据四、配置优化1. 调整维护工作负载2. 监控和日志 五、使用外部存储和扩展1. 外部表和FDW2. 扩展和插件 六、定期备份和恢复测试结论 管理大型数据库的大小和增长是数据库管理员&#x…

EA包图上嵌套的包位置不对

Extreme 2024-4-11 11:36 我从工具栏把一个包拖在另一个包里面&#xff0c;可是项目树上两个包的位置并列&#xff0c;拖了几次结果都一样。我的目的是做一个多层级的包图&#xff0c;是不是&#xff08;EA&#xff09;不能在图上做&#xff1f; UMLChina潘加宇 确实是这样&a…

m个人的成绩存放在score数组中,请编写函数fun,它的功能是:将低于平均分的人数作为函数值返回,将低于平均分的分数放在below所指的数组中。

本文收录于专栏:算法之翼 https://blog.csdn.net/weixin_52908342/category_10943144.html 订阅后本专栏全部文章可见。 本文含有题目的题干、解题思路、解题思路、解题代码、代码解析。本文分别包含C语言、C++、Java、Python四种语言的解法和详细的解析。 题干 m个人的成绩…

强固型工业电脑在轮胎成型机设备行业应用

轮胎成型机设备行业应用 项目背景 我国是一个轮胎生产与出口大国&#xff0c;在轮胎的生产过程中&#xff0c;成型工序是生产的关键工序&#xff0c;将半成品的部件按照轮胎的技术规范在成型机上组装成胎胚&#xff0c;工艺较为复杂&#xff0c;对设备加工和定位精度要求较高。…

【Linux学习】gcc与g++的使用与程序翻译的过程

文章目录 修改sudo用户名单gcc与g指令gcc用法g用法 程序翻译的过程函数库动态库与静态库什么是动静态库&#xff1f;动静态库的优缺点ldd指令 这里补充一个前面遗漏的知识&#xff1a; 修改sudo用户名单 必须使用root超级用户进行修改&#xff0c;普通用户是不能访问修改。 …

Spring 事务实现方式:

Spring 事务实现方式&#xff1a; Spring并不直接支持事务&#xff0c;只有当数据库支持事务的时候&#xff0c;Spring才支持事务&#xff0c;Spring只不过简化了开发人员实现事务的开发步骤 Spring事务的实现方式有两种&#xff1a; 一、基于申明式事务&#xff1a; Service…