Java的异常处理

news2025/2/24 7:38:58

t656TWPNb10KQ0GT26iy

异常

异常就是程序非正常运行时的报错,不正常就是异常。

异常分类

通常分为两类:

Error:错误。通常是Java虚拟机无法解决的严重问题。如:JVM系统内部错误、资源耗尽等严重情况。比如:StackOverflowErrorOOM-->OutOfMemory

一般不编写针对性的代码进行处理。

Exception:其它因编程错误或偶然的外在因素导致的一般性问题, 可以使用针对性的代码进行处理。

异常体系结构

父类为Throwable

图中蓝色的是非受检异常(unchecked)或称运行时异常。

红色是受检异常(checked)或称编译时异常,会在编译时进行报错。

异常体系架构

常见异常

Exception

编译时异常(checked)

  • IOException
    • FileNotFoundException
  • ClassNotFoundException

运行时异常(unchecked,RuntimeException)

  • NullPointerException–>空指针异常

  • ArrayIndexOutOfBoundsException–>数组越界异常

  • ClassCastException–>类转换异常

  • NumberFormatException–>数值格式异常

  • InputMismatchException–>输入不匹配异常

  • ArithmeticException–>算术异常

    示例:

    import java.util.Scanner;
    import org.junit.Test;
    
    public class ExceptionTest {
        @Test
        public void test1(){
            //NullPointerException
            int[] arr = new int[5];
            arr = null;
            System.out.println(arr[0]);
        }
    
        @Test
        public void test2(){
            //ArrayIndexOutOfBoundsException
            int[] arr = new int[]{1,2,3};
            System.out.println(arr[3]);
        }
    
        @Test
        public void test3(){
            //ClassCastException
            Object s = "abc";
            int i = (int)s;
            
        }
    
        @Test
        public void test4(){
            //NumberFormatException
            String s = "123";
            int i = Integer.parseInt(s);
            System.out.println(i);
        }
    
        @Test
        public void test5(){
            //InputMismatchException
            System.out.println("please input");
            Scanner scanner = new Scanner(System.in);
            int i = scanner.nextInt();
            System.out.println(i);
            scanner.close();
        }
    
        @Test
        public void test6(){
            //ArithmeticException
            System.out.println(2 / 0);
    
        }
    }
    

异常处理

抓抛模型

过程一:“抛”:程序在正常执行的过程中,一旦出现异常,就会在异常代码处生成一个对应异常类的对象。并将此对象抛出。

一旦抛出对象以后,其后的代码就不再执行。

过程二:“抓”:可以理解为异常的处理方式:

  1. try-catch-finally

  2. throws

try-catch-finally

这个结构可以抓住当前异常进行处理。

try{
	//可能出现异常的代码
}catch(异常类型1 变量名1){
	//处理异常的方式1
}catch(异常类型2 变量名2){
	//处理异常的方式2
}catch(异常类型3 变量名3){
	//处理异常的方式3
}
....
finally{
	//一定会执行的代码
}
  1. finally是可选的。

  2. 使用try将可能出现异常代码包装起来,在执行过程中,一旦出现异常,就会生成一个对应异常类的对象,根据此对象的类型,去catch中进行匹配

  3. 一旦try中的异常对象匹配到某一个catch时,就进入catch中进行异常的处理。一旦处理完成,就跳出当前的try-catch结构(在没有写finally的情况)。继续执行其后的代码

  4. catch中的异常类型如果没有子父类关系,则谁声明在上,谁声明在下无所谓。

    catch中的异常类型如果满足子父类关系,则要求子类一定声明在父类的上面。否则,会报错

  5. 常用的异常对象处理的方式:

    • String getMessage()
    • printStackTrace()
  6. 在try结构中声明的变量,再出了try结构以后,就不能再被调用

  7. try-catch-finally结构可以嵌套

体会

  1. 使用try-catch-finally处理编译时异常,使得程序在编译时不再报错,但是运行时仍可能报错。相当于我们使用try-catch-finally将一个编译时可能出现的异常,延迟到运行时出现。

  2. 开发中,由于运行时异常比较常见,所以我们通常就不针对运行时异常编写try-catch-finally了

​ 而针对于编译时异常,我们说一定要考虑异常的处理。(对于这一点,你可以先往下看,到finally结束时再次回顾。)

finally的二三事

  1. finally中声明的是一定会被执行的代码。即使catch中又出现异常了,或者try中有return语句,catch中有return语句等情况下,也会被强制在方法返回前执行。
  2. 像数据库连接、输入输出流、网络编程Socket等资源,JVM是不能自动的回收的,我们需要自己手动的进行资源的释放。此时的资源释放,就需要声明在finally中。
@Test
    public void test1(){
        FileInputStream fis = null;
        try {
            File file = new File("hello.txt");
            //当文件存在,程序正常运行;文件不存在,打印异常,两种情况finally都会执行。
            fis = new FileInputStream(file);

            int data  = fis.read();
            while (data != -1) {
                System.out.println((char)data);
                data = fis.read();
            }
            
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            try {
                System.out.println("程序结束啦,现在finally。。。");
                //如果是异常到这里,fis应该是空的,所以要判断一下
                if (fis != null) {
                    fis.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

写完这个程序你会发现,如果是编译时异常,我们使用try-catch将其延长到运行时才进行报错,或者说,程序强制要求我们进行处理,就拿文件处理来说,我们预见一定会有找不到文件的情况,并且要求处理。而如果是运行时异常,则程序并不报错,只有运行起来我们才知道有bug。

这种情况下,我们会发现,捕捉了运行时异常并进行报错,实际和程序自己进行报错没什么区别。所以我们才不针对运行时异常编写try-catch-finally(一般情况下)

throws

这个结构只能将异常进行抛出。不能处理。

public void method() throws IOException{ }

"throws + 异常类型"写在方法的声明处。指明此方法执行时,可能会抛出的异常类型。一旦当方法体执行时,出现异常,仍会在异常代码处生成一个异常类的对象,此对象满足throws后异常类型时,就会被抛出。异常代码后续的代码,就不再执行!

二者的区别

try-catch-finally:真正的将异常给处理掉了。

throws的方式只是将异常抛给了方法的调用者。 并没有真正将异常处理掉。

如何选择

  1. 如果父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws,意味着如果子类重写的方法中有异常,必须使用try-catch-finally方式处理。

  2. 执行的方法a中,先后又调用了另外的几个方法,这几个方法是递进关系执行的。我们建议这几个方法使用throws的方式进行处理。而执行的方法a可以考虑使用try-catch-finally方式进行处理。

手动抛出异常

生成官方异常对象

throw关键字生成异常对象

public class Student {
    private int id;

    public void regester(int id){
        if (id > 0) {
            this.id = id;
        } else {
            throw new RuntimeException("您输入的数据非法!");
            //自己抛出一个异常,并且传入错误信息
        }
    }

    @Override
    public String toString() {
        return "Student [id=" + id + "]";
    }
    
    public static void main(String[] args) {
        Student s1 = new Student();
        s1.regester(-1);
        System.out.println(s1);
    }
}

自定义异常类型

分三步:

  1. 继承异常体系
  2. 给出唯一标识serialVersionUID
  3. 提供重载的构造器
public class MyException extends RuntimeException{
    static final long serialVersionUID = -338751699363429948L;

    public MyException() {
        super();
    }

    public MyException(String message) {
        super(message);
    }
}

之后当作正常的异常进行抛出就行了

练习

编写应用程序EcmDef.java,接收命令行的两个参数,要求不能输入负数,计算两数相除。

对数据类型不一致NumberFormatException、缺少命令行参数ArrayIndexOutOfBoundsException、除0ArithmeticException及输入负数EcDef 自定义的异常进行异常处理。

提示:

  1. 在主类(EcmDef)中定义异常方法(ecm)完成两数相除功能。
  2. 在main()方法中使用异常处理语句进行异常处理。
  3. 在程序中,自定义对应输入负数的异常类(EcDef)。
  4. 运行时接受参数 java EcmDef 20 10 //args[0]=“20” args[1]=“10”
  5. Interger类的static方法parseInt(String s)将s转换成对应的int值。

如:int a=Interger.parseInt(“314”); //a=314;

public class EcmDef {
    
    public static void main(String[] args) {
        try {
            int i = Integer.parseInt(args[0]);
            int j = Integer.parseInt(args[1]);
            System.out.println(ecm(i, j));
        } catch (NumberFormatException e) {
            System.out.println("数值转换异常!");
        } catch (ArrayIndexOutOfBoundsException e) {
            System.out.println("缺少参数!");
        } catch (ArithmeticException e) {
            System.out.println("除零!");
        } catch (EcDef e) {//如果是编译时异常,就要加上处理语句
            System.out.println(e.getMessage());
        }
    }
    
    public static int ecm(int i,int j){//加上throws EcDef
        if (i < 0 | j < 0) {
            throw new EcDef("请不要输入负数");
        }
        return i/j;
    }
}
public class EcDef extends RuntimeException{//若继承Exception,就是编译时异常
    static final long serialVersionUID = -70348972321766939L;
    public EcDef() {}
    public EcDef(String message) {super(message);}
}

我这里写的是一个运行时异常,所以没有进行异常处理。

但如果你写的是编译时异常,就要选择进行处理

总结

image-20230128142744628

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

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

相关文章

canvas根据坐标点位画图形-canvas拖拽编辑单个图形形状

首先在选中图形的时候需要用鼠标右击来弹出选择框&#xff0c;实现第一个编辑节点功能 在components文件夹下新建右键菜单 RightMenu文件&#xff1a; <template><div v-show"show" class"right-menu" :style"top:this.ypx;left:this.xpx…

PCB设计中的数据单位,层作用,工艺要求,制作过程等简介

PCB中常识 物理结构常识 几乎每层板子都是由两个金属层夹着一个绝缘层&#xff0c;像一个三明治结构。多层板是这种双层板叠加压合的。 常见的层的作用 参考&#xff1a;https://www.21ic.com/article/854059.html PCB电路板一般而言会分成六层&#xff0c;具体为&#x…

第二十四章 java并发常见知识内容(AQS)

JAVA重要知识点AQS介绍AQS的核心思想AQS资源共享方式自定义同步器常见同步工具类Semaphore(信号量)CountDownLatch(倒计时器)CyclicBarrier(循环栅栏)AQS 介绍 AQS 的全称为 AbstractQueuedSynchronizer &#xff0c;翻译过来的意思就是抽象队列同步器。这个类在 java.util.c…

图解 git 常用命令

本文图解 Git 中的最常用命令&#xff0c;如果你稍微理解 Git 的工作原理&#xff0c;这篇文章能够让你理解的更透彻。 基本用法 上面的四条命令在工作目录、暂存目录(也叫做索引)和仓库之间复制文件。 git add files 把当前文件放入暂存区域。 git commit 给暂存区域生成快照…

java集合:Collection:Set+List+Dueue

目录 Collection集合 1.常用方法 2.使用Iterator接口遍历集合元素 3.使用foreach循环遍历集合元素 4.Set集合 4.1 HashSet类 4.2 LinkedHashSet类 4.3 TreeSet类 4.4 EnumSet类 5. List接口 5.1 ListIteractor接口 5.2 Stack类 5.3 Arrays.ArrayList 5.4 Linke…

ChatGPT体验失败的经历

ChatGPT的介绍 最近有一个很热门的话题&#xff0c;而且这个话题内容已经覆盖了多个领域。ChatGPT这个词&#xff0c;最近不断的出现在微博的热搜或者公众号的文章里。那我们先大概了解一下ChatGPT。 ChatGPT是由OpenAI开发的一个人工智能聊天机器人程序&#xff0c;于2022年11…

在Allegro设计界面中如何修改封装焊盘

在Allegro PCB设计界面中修改封装焊盘的方法1、选择菜单Tools→Padstack→Modify Design Padstack...2、然后点击封装上要修改的焊盘&#xff0c;右击选择Edit或者点击Options选项卡下要修改的焊盘&#xff0c;然后点击Edit修改3、修改完焊盘后&#xff0c;选择File→Update to…

PyTorch学习笔记:nn.PReLU——PReLU激活函数

PyTorch学习笔记&#xff1a;nn.PReLU——PReLU激活函数 torch.nn.PReLU(num_parameters1, init0.25, deviceNone, dtypeNone)功能&#xff1a;逐元素对数据应用如下函数公式进行激活 PReLU(x)max⁡(0,x)a∗min⁡(0,x)\text{PReLU}(x)\max(0,x)a*\min(0,x) PReLU(x)max(0,x)a∗…

JS document.write()换行

换行效果&#xff1a; 通过传递多个参数&#xff0c;即可实现换行效果&#xff1a; document.write("<br>",ar) 效果&#xff1a; 示例源码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8&quo…

Vue3 中 导航守卫 的使用

目录前言&#xff1a;一、什么是导航守卫&#xff1f;二、导航守卫有哪几种&#xff1f;1、全局守卫&#xff08;3个&#xff09;全局前置守卫全局解析守卫全局后置钩子2、路由独享守卫&#xff08;1个&#xff09;3、组件内守卫&#xff08;3个&#xff09;beforeRouteEnter(t…

大数据Kylin(二):Kylin安装使用

文章目录 Kylin安装使用 一、Kylin安装要求 二、Kylin安装 1、Kylin安装前环境准备

SpringCloud-Netflix学习笔记12——Hystrix-Dashboard实现服务监控

一、服务监控 hystrixDashboard 除了隔离依赖服务的调用以外&#xff0c;Hystrix还提供了准实时的调用监控&#xff08;Hystrix Dashboard&#xff09;&#xff0c;Hystrix会持续地记录所有通过Hystrix发起的请求的执行信息&#xff0c;并以统计报表和图形的形式展示给用户&…

MikTex+Texmaker环境LaTeX使用bib文件引用参考文献步骤

一、MikTexTexmaker环境 MikTexTexmaker环境是一个LaTeX编辑与编译&#xff0c;以及PDF预览环境。如果还没装好的话&#xff0c;推荐浏览【MiKTeXTexmaker安装】这篇博客去安装此环境。 二、准备bib文件 文件名命名为ref.bib&#xff0c;添加如下内容(作为demo) article{kaelb…

微信小程序nodejs+java+python美食制作教程系统

食谱大全所要实现的功能分析,对于现在网络方便,食谱大全要实现管理员、用户可以直接在平台上进行查看自己所需数据信息,这样既能节省管理的时间,不用再像传统的方式,如果用户想要进行交流信息,必须双方见面进行沟通交流所需的信息,由于很多用户时间的原因,没有办法进行见面沟通…

MySQL_InnoDB引擎

InnoDB引擎 逻辑存储结构 表空间&#xff08;ibd文件&#xff09;&#xff0c;一个mysql实例可以对应多个表空间&#xff0c;用于存储记录、索引等数据。 段&#xff0c;分为数据段&#xff08;Leaf node segment&#xff09;、索引段(Non-leaf node segment)、回滚段(Rollba…

【LeetCode】每日一题(1)

目录 题目&#xff1a; 解题思路&#xff1a; 代码&#xff1a; 写在最后&#xff1a; 题目&#xff1a; 这是他给出的接口&#xff1a; class Solution { public:int fillCups(vector<int>& amount) {} }; 作为一个数学学渣&#xff0c;我想不出厉害的数学算法…

151、【动态规划】AcWing ——2. 01背包问题:二维数组+一维数组(C++版本)

题目描述 原题链接&#xff1a;2. 01背包问题 解题思路 &#xff08;1&#xff09;二维dp数组 动态规划五步曲&#xff1a; &#xff08;1&#xff09;dp[i][j]的含义&#xff1a; 容量为j时&#xff0c;从物品1-物品i中取物品&#xff0c;可达到的最大价值 &#xff08;2…

ChatGPT入门系列(一)——注册

ChatGPT入门教程最近一段时间&#xff0c;OpenAI发布的ChatGPT聊天机器人太火了&#xff0c;连着上了各个平台的热搜榜。这个聊天机器人最大的特点是模仿人类说话风格&#xff0c;同时回答各种问题。有人说ChatGPT是真正的人工智能&#xff0c;它不仅能和你聊天&#xff0c;还是…

我的企业需要一个网站吗?答案是肯定的 10 个理由

如果您的企业在没有网站的情况下走到了这一步&#xff0c;您可能会想&#xff1a;我的企业需要一个网站吗&#xff1f;如果我的企业没有一个就已经成功了&#xff0c;那又有什么意义呢&#xff1f;简短的回答是&#xff0c;现在是为您的企业投资网站的最佳或更重要的时机。网站…

C/C++排序算法(二) —— 选择排序和堆排序

文章目录前言1. 直接选择排序&#x1f351; 基本思想&#x1f351; 具体步骤&#x1f351; 具体步骤&#x1f351; 动图演示&#x1f351; 代码实现&#x1f351; 代码升级&#x1f351; 特性总结2. 堆排序&#x1f351; 向下调整算法&#x1f351; 任意树调整为堆的思想&#…