Java中的异常语法知识居然这么好玩!后悔没有早点学习

学习异常后,发现异常的知识是多么的吸引人!不仅可以用来标记错误,还可以自己定义一个异常,用来实现自己想完成的业务逻辑,接下来一起去学习吧

目录

一、异常的概念及体系结构

1.异常的概念

2.异常的体系结构

3.异常的分类 

二、怎么处理异常

1.防御式编程

2.异常的抛出

3.异常的捕获

4.异常的处理流程

三、自定义异常

1.自定义异常的用途及注意事项

2. 自定义异常实现登录场景


一、异常的概念及体系结构

1.异常的概念

(1)什么是异常

所谓异常,就是程序在执行的过程中,发生的不正常行为;也可以认为是代码存在bug

(2)常见的异常

  • 算术异常

也就是计算的过程中发生的异常,如分目不能为0,但是硬要写成0

 public static void main(String[] args) {
     //算术异常
        System.out.println(10/0);
        System.out.println("检验这句话是否被打印");
    }

  • 数组越界异常

常见的如:对数组越界访问

 public static void main(String[] args) {
        //数组越界异常
        int[] arr = {1,2,3,4,5};
        System.out.println(arr[6]);
        System.out.println("检验这句话是否被打印");
    }

  • 空指针异常

空指针异常,并不等于Java中存在指针,而是对空引用的变量进行访问

public static void main(String[] args) {
        //空指针异常
        int[] arr = null;//此时arr为一个空引用
        System.out.println(arr[10]);
        System.out.println("检验这句话是否被打印");
    }

(3)异常小总结

  • 当程序发生异常之后,程序就被终止了,后面的代码不会再被执行
  • 发生异常前的代码依旧可以正常执行
 public static void main(String[] args) {
        //空指针异常
        System.out.println("在异常发生前,是否会打印这句话");

        int[] arr = null;//此时arr为一个空引用
        System.out.println(arr[10]);
        System.out.println("检验这句话是否被打印");
    }

2.异常的体系结构

(1)每一个异常都是一个类,它们之间的关系为继承

(2)异常的体系结构

  • Error:指的是Java虚拟机无法解决的严重问题,JVM的内部错误、资源耗尽等,如:栈溢出(StackOverflowError)和内存不足错误(OutOfMemoryError)

下面简单举个例子:无限递归致使栈溢出

 public static void main(String[] args) {
        func();
    }
    public static void func() {
        func();//无限递归
    }

3.异常的分类 

这里的异常,也就是指Exception,后面产生的两个子类

(1)编译时异常(Checked Exception)--受查异常

编译时异常就是在编写代码的时候就报的错误,下面举一个克隆异常的例子

这是不支持克隆的异常,怎么做?需要声明异常:鼠标放到异常处,Alt+Enter键即可(要想实现克隆,还需要实现克隆接口)

 (2)运行时异常(RutimeException)--非受查异常

在程序执行期间发生的异常,称为运行时异常,也称为非受检查异常(Unchecked Exception)

如:RunTimeException以及其子类对应的异常,都称为运行时异常

二、怎么处理异常

异常处理主要的5个关键字:throw、try、catch、finaly、throws

1.防御式编程

(1)LBYL(事前防御型)

  • 也就是在走每一步前需要先确认有没有错误发生,A->B->C->D的走法,
  • 模板
boolean ret = false;
ret = 登陆游戏();
if (!ret) {
    处理登陆游戏错误;
    return;
}
ret = 开始匹配();
if (!ret) {
处理匹配错误;
  return;
}
ret = 游戏确认();
if (!ret) {
处理游戏确认错误;
  return;
}
ret = 选择英雄();
if (!ret) {
  处理选择英雄错误;
  return;
}
ret = 载入游戏画面();
if (!ret) {
处理载入游戏错误;
  return;
}

登录游戏时就确认有没有错误,正确才能进入下一步

  • 缺点:正常流程和错误处理流程代码混在一起, 代码整体显的比较混乱

(2)EAFP(事后认错型)-主推

       EAFP是异常处理的核心思想

  • 先把代码给执行了,再去判断有没有异常
  • 模板
try {
  登陆游戏();
  开始匹配();
  游戏确认();
  选择英雄();
  载入游戏画面();
 ...
} catch (登陆游戏异常) {
  处理登陆游戏异常;
} catch (开始匹配异常) {
处理开始匹配异常;
} catch (游戏确认异常) {
处理游戏确认异常;
} catch (选择英雄异常) {
处理选择英雄异常;
} catch (载入游戏画面异常) {
处理载入游戏画面异常;
}
  • 优点: 正常流程和错误流程是分离开的, 程序员更关注正常流程,代码更清晰,容易理解代码

2.异常的抛出

(1)理解throw

  • 异常的抛出需要借助关键字:throw
  • 格式:
    throw new XXXException("异常产生的原因");

    也就是throw一个异常的对象,给对象的传参可以自己定义

  • 作用:一般用于标记可能发生的异常;当异常发生时,程序员可以快速定位发生异常的地方;一般会用于抛出程序员自定义的异常(需要程序员自己定义某个异常的类)
  • 举例说明:
public static void func(int[] arr) {
        if(arr == null) {
            throw new NullPointerException("这是自己抛出的空指针异常");
        }
    }
    public static void main(String[] args) {
        func(null);
    }

(2)throw的注意事项

  • throw必须写在方法体内部

  • 抛出的对象必须是Exception 或者 Exception 的子类对象
  • 如果抛出的是 RunTimeException 或者 RunTimeException 的子类,则可以不用处理,直接交给JVM来处理
  • 如果抛出的是编译时异常,用户必须处理,否则无法通过编译
  • 异常一旦抛出,其后的代码就不会执行

3.异常的捕获

(1)异常的声明

  • 异常的声明需要用到关键字:throws

异常声明的位置:处在方法声明时参数列表之后,可以同时声明多个异常

作用:当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛
给方法的调用者来处理。即当前方法不处理异常,提醒方法的调用者处理异常

格式:

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

(2)注意事项

  • throws必须跟在方法的参数列表之后
  • 声明的异常必须是 Exception 或者 Exception 的子类
  • 方法内部如果抛出了多个异常,throws之后必须跟多个异常类型,之间用逗号隔开,如果抛出多个异常类型具有父子关系,直接声明父类即可。

比如:声明的多个异常类型都是运行时异常的子类,那么可以直接声明运行时异常(RuntimeExcepetion),更极端可以直接声明异常(Exception)

  • 调用声明抛出异常的方法时,调用者必须对该异常进行处理,或者继续使用throws抛出

举例:

当被调用的方法(func)声明了异常之后:

说明:当方法后面直接声明Exception时,可能是运行时异常。也有可能是编译使异常;当什么都没有时,会默认是编译时异常,所以就报错了。

解决报错的第一种方法:调用者所在的方法也要声明同样的异常

解决报错的第二种方法:对可能发生异常的代码进行捕获,也就是“异常的捕获”;使用try{}catch对异常捕获:

 

也就是下面的内容

(2)try-catch捕获并处理

【简单语法】

正常的对异常进行声明或者抛出,只是简单的介绍了或者只是处理了编译时异常,而未正在的处理过异常;当异常不处理时,最后会把异常交给JAVM处理,则程序便会终止,不再执行

  • 简单try-catch的语法

简单举例:只要捕获到异常,代码就可以继续往下执行

 public static void func(){
        int[] arr = null;
        System.out.println(arr.length);
    }
    public static void main(String[] args) {
        try{
            func();
        }catch (NullPointerException e) {
            System.out.println("处理NullPointerException异常成功!");
        }
        System.out.println("处理完异常可以继续走完下面的代码!");
    }

【注意事项】

  • try块内抛出异常位置之后的代码将不会被执行

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

  • 一个catch可以捕获多个异常(不推荐)

 

  • 可以有多个catch捕获异常(推荐)
 public static void main(String[] args) {
        try{
            int[] arr = {1,2,3};
            System.out.println(arr[10]);
        }catch (NullPointerException e) {
            e.printStackTrace();
            System.out.println("成功捕获到空指针异常");
        }catch(ArrayIndexOutOfBoundsException e) {
            e.printStackTrace();
            System.out.println("捕获到了数组越界异常");
        } 
        System.out.println("后续代码可被执行!");
    }

  • 可以通过一个catch捕获所有的异常(不推荐)

  • 如果异常之间具有父子关系,一定是子类异常在前catch,父类异常在后catch,否则语法错误

做法:子类必须在父类前面(否在永远不会执行到子类,父类在后可以兜底)

【引入finally】

  • 语法特点:
try{
    // 可能会发生异常的代码
}catch(异常类型  e) {
    //对捕获到的异常进行处理
}catch(异常类型  e) {
    //对捕获到的异常进行处理
}finally {
    //此处的语句无论是否发生异常,都会被执行到
}

【finally特点】

  • finally中的语句一定会被执行,多用于因为某些情况(如:程序异常退出,文件保存等)没有执行到的代码,可以放在finally中,完成一个兜底的作用
  • 自动完成对资源的关闭

  • 直接在try后面接小括号
public static void main(String[] args) {
        
        try(Scanner scanner = new Scanner(System.in)){
            int a = scanner.nextInt();
        }
        catch (NullPointerException e){
            System.out.println("子类在前");
        }catch (RuntimeException e) {
            System.out.println("父类在后");
        }finally {
            System.out.println("finally被执行了哦!");
        }
    }
  • 输入异常

做法: 捕获异常即可

  • finally与return
 public static int func6() {
        try(Scanner scanner = new Scanner(System.in)){
            int a = scanner.nextInt();
            return a;
        }
        catch (NullPointerException e){
            e.printStackTrace();
            System.out.println("子类在前");
        }finally {
            System.out.println("finally被执行了哦!");
            return -1;
        }
       
    }
    public static void main(String[] args) {
        int ret = func6();
        System.out.println("接收的返回值"+ret);
    }

输入:20

总结:finally中的语句一定会被执行,即使前面存在return;存在多个return,最终结果以finally中的为准

4.异常的处理流程

(1)简单三部曲

方法中是否有处理异常(未处理则下一步)--->调用该方法有没有处理异常(未处理则交给JVM)--->JVM最后处理异常,程序则会终止

 public static void main(String[] args) {
        func7();//调用者也未处理该异常
    }
    
    public static void func7() {
        try{
            int[] arr = {1,2,3};
            System.out.println(arr[5]);//数组越界访问异常
        }
        catch (NullPointerException e){
            e.printStackTrace();
            System.out.println("子类在前");
        }finally {
            System.out.println("finally被执行了哦!");
        }

    }

做法:

 public static void main(String[] args) {
        try {
            func7();//在调用者处处理异常
        }catch (ArrayIndexOutOfBoundsException e) {
            e.printStackTrace();
        }
    }

    public static void func7() {
        try{
            int[] arr = {1,2,3};
            System.out.println(arr[5]);//数组越界访问异常
        }
        catch (NullPointerException e){
            e.printStackTrace();
            System.out.println("子类在前");
        }finally {
            System.out.println("finally被执行了哦!");
        }

    }

(2)总结流程

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

三、自定义异常

1.自定义异常的用途及注意事项

(1)多用于一些业务逻辑中可能发生的异常,常常是系统中没有的;如账号登录时,输入的账号错误或者密码错误等等

(2)自定义的异常一般继承RuntimeException或者Exception

2. 自定义异常实现登录场景

(1)先实现一个登录逻辑的类

public class Logic {
    public String userName = "zhangsan";//设置初始账号名字为:zhangsan
    public String password = "123456";//初始密码为1234456

    public void  loginInfo(String userName,String password) {
        //该方法用来验证密码是否正确,参数为
        //比较用户名
        if(!this.userName.equals(userName)) {
            System.out.println("用户名错误!");
        }
        //比较密码
        if(!this.password.equals(password)) {
            System.out.println("密码错误!");
        }
    }
}

以上是账户登录的大概逻辑 

(2)实现自定义异常类 

该类用来:当账户或密码错误时,抛出异常信息并定位错误的行号,利于修改

用户名异常类:

public class UerNameException extends RuntimeException{
    //用户异常类
    public UerNameException() {
        super();
    }

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

密码异常类:

public class PassWordException extends RuntimeException{
    //账号密码错异常类
    
    //模拟原码实现两个构造方法
    public PassWordException() {
        super();
    }
    
    public PassWordException(String s) {
        super(s);
    }
}

(3)完成业务逻辑

为加异常类时:

public class Logic {
    public String userName = "zhangsan";//设置初始账号名字为:zhangsan
    public String password = "123456";//初始密码为1234456

    public void  loginInfo(String userName,String password) {
        //该方法用来验证密码是否正确,参数为
        //比较用户名
        if(!this.userName.equals(userName)) {
            System.out.println("用户名错误!");
        }
        //比较密码
        if(!this.password.equals(password)) {
            System.out.println("密码错误!");
        }
    }

    public static void main(String[] args) {
        Logic logic = new Logic();
        logic.loginInfo("lisi","6666");//调用方法输入账号和密码


    }
}

注入异常后:

public class Logic {
    public String userName = "zhangsan";//设置初始账号名字为:zhangsan
    public String password = "123456";//初始密码为1234456

    public void  loginInfo(String userName,String password) {
        //该方法用来验证密码是否正确,参数为
        //比较用户名
        if(!this.userName.equals(userName)) {
            //System.out.println("用户名错误!");
            throw new UerNameException("用户名错误!");//抛出异常
        }
        //比较密码
        if(!this.password.equals(password)) {
            //System.out.println("密码错误!");
            throw new PassWordException("密码错误!");//抛出异常
        }
    }

    public static void main(String[] args) {
        try {
            Logic logic = new Logic();
            logic.loginInfo("lisi","6666");//调用方法输入账号和密码
        }catch (UerNameException e) {
            e.printStackTrace();
        }catch (PassWordException e) {
            e.printStackTrace();
        }
        System.out.println("捕获异常后,不影响代码继续往下执行!");
    }
}

这里用户名错误后,不再判断密码 

测试密码错误:

(4)一些小问题

当自定义异常继承Exception需要添加的细节

 因为Exception默认是受查异常/编译时异常,所以需要加上解决掉报错


本次的内容分享到这里就结束了,小伙伴快去试试吧!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【数据处理】 -- 【两分钟】了解【最好】的方式 -- 【正则表达式】

直接匹配; 普通字符 元匹配: . 任意单字符 r’表示单引号里字符为其特殊含义,比如.不是句号是匹配符的意思 *任意次数(换行结束) 一次及以上 {3,4}指定次数,至少3次,最多4次|{3}固定4次 [\d.]单个任意…

软件工程简明教程

软件工程简明教程 何为软件工程? 1968 年 NATO(北大西洋公约组织)提出了软件危机(Software crisis)一词。同年,为了解决软件危机问题,“软件工程”的概念诞生了。一门叫做软件工程的学科也就应…

redis运维(二十)redis 的扩展应用 lua(二)

一 redis 的扩展应用 lua redis lua脚本语法 ① 什么是脚本缓存 redis 缓存lua脚本 说明: 重启redis,脚本缓存会丢失 下面讲解 SCRIPT ... 系列 SCRIPT ② LOAD 语法:SCRIPT LOAD lua代码 -->载入一个脚本,只是预加载,不执行思考1&#xff1…

吴恩达《机器学习》10-4-10-5:诊断偏差和方差、正则化和偏差/方差

一、诊断偏差和方差 在机器学习中,诊断偏差和方差是改进模型性能的关键步骤。通过了解这两个概念,能够判断算法的问题究竟是欠拟合还是过拟合,从而有针对性地调整模型。 1. 概念理解 偏差(Bias): 表示模…

《微信小程序开发从入门到实战》学习三十一

3.4 开发参与投票页面 3.4.9 显示投票结果 在实际使用中,一个用户不能对同一个投票进行重复提交,因此需要向服务器端提交投票结果和提交用户ID。另外页面,需要完善。用户提交完投票后 ,还需要显示投票目前的结果,提交…

C#,《小白学程序》第二十课:大数的加法(BigInteger Add)

大数的&#xff08;加减乘除&#xff09;四则运算、阶乘运算。 乘法计算包括小学生算法、Karatsuba和Toom-Cook3算法。 重复了部分 19 课的代码。 1 文本格式 using System; using System.Linq; using System.Text; using System.Collections.Generic; /// <summary>…

字符串函数

目录 读取字符串的函数 1.gets()函数 2.fgets()函数&#xff08;不是所有的编译器都支持例如CodeBlocks&#xff09; 3.scanf()函数 4.getchar()函数 输出字符串的函数 1.puts()函数 2.fputs()函数&#xff08;编译器不一定支持&#xff09; 3.printf()函数 4.putchar…

【开源】基于Vue.js的陕西非物质文化遗产网站

文末获取源码&#xff0c;项目编号&#xff1a; S 065 。 \color{red}{文末获取源码&#xff0c;项目编号&#xff1a;S065。} 文末获取源码&#xff0c;项目编号&#xff1a;S065。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 设计目标2.2 研究内容2.3 研究方法与…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《基于多尺度分量特征学习的用户级超短期负荷预测》

这篇文章的标题表明研究的主题是用户级超短期负荷预测&#xff0c;并且该预测方法基于多尺度分量特征学习。让我们逐步解读这个标题&#xff1a; 用户级&#xff1a; 这表示研究的焦点是在个体用户层面上进行的。负荷预测可能是指电力系统中的负荷&#xff0c;即电力需求。用户…

大模型能否生成搜索引擎的未来?

文&#xff5c;郝 鑫 编&#xff5c;刘雨琦 ChatGPT火爆之前&#xff0c;水面下&#xff0c;也有中国公司也在朝着智能助手的方向努力。夸克便是其中之一。在GPT风靡科技圈后&#xff0c;国内就开始陆续冒出一些大模型厂商。对当时夸克而言&#xff0c;做大模型毋庸置疑&am…

五种多目标优化算法(MOPSO、MOAHA、NSGA2、NSGA3、MOGWO)求解微电网多目标优化调度(MATLAB)

一、多目标优化算法简介 &#xff08;1&#xff09;多目标粒子群优化算法MOPSO 多目标应用&#xff1a;基于多目标粒子群优化算法MOPSO求解微电网多目标优化调度&#xff08;MATLAB代码&#xff09;-CSDN博客 &#xff08;2&#xff09;多目标人工蜂鸟算法&#xff08;MOAHA…

Redis-Redis 高并发分布式锁

集群分布式场景高并发 1.negix配置代理和路由 高并发场景超卖问题 1.使用原生redis控制超卖时(若是商品&#xff0c;则可以将商品id作为锁对象)&#xff0c;会遇到的问题 问题一&#xff1a;若直接使用&#xff1a;将获取锁的对象和设置的超时的时间分开&#xff0c;则不能控…

桥接设计模式

package com.jmj.pattern.bridge;/*** 视频文件(实现化角色)*/ public interface VideoFile {void decode(String fileName); }package com.jmj.pattern.bridge;public class RmvFile implements VideoFile{Overridepublic void decode(String fileName) {System.out.println(&…

论文阅读——MCAN(cvpr2019)

补充一下MCAN-VQA&#xff1a; 对图片的处理&#xff1a;首先输入图片到Faster R-CNN&#xff0c;会先设定一个判断是否检测到物体的阈值&#xff0c;这样动态的生成m∈[10,100]个目标&#xff0c;然后从检测到的对应的区域通过平均池化提取特征。第i个物体特征表示为&#xff…

ubuntu22.04系统下载程序和依赖,并拷贝到指定路径下

脚本1 apt install aptitude apt-get -d install xxx #xxx是待下载的安装包 mv /var/cache/apt/archives/* /home/tuners/1apt install aptitude apt-get -d install xxx mv /var/cache/apt/archives/*.deb /home/tuners/1 xxx 为程序包名称 /home/tuners/1为保存程序包的…

网络通信基础概念介绍

网络通信基础概念介绍 局域网LAN 局域网&#xff0c;即 Local Area Network&#xff0c;简称LAN。 局域网内的主机之间能方便的进行网络通信&#xff0c;又称为内网&#xff1b;局域网和局域网之间在没有连接的情况下&#xff0c;是无法通信的。 局域网是指在一个相对较小的…

微机课设--汇编语言在51单片机上写一个四位十进制加法器

代码如下 KEYVAL EQU 30HKEYTM EQU 31HKEYSCAN EQU 32HDAT EQU 33HSCANLED EQU 37HS_DAT EQU 38HD_DAT EQU 39HR_DATL EQU 3AHR_DATH EQU 3BH CALFLAG EQU 3CHFLAG BIT 00HORG 0000HLJMP MAINORG 000BHLJMP T0ISRORG 0030HMAIN:MOV SP,#5FHMOV TMOD,#01HMOV TH0,#0D8HMOV TL0,…

过渡曲线的构造之平面PH曲线

平面PH曲线的构造及其相应性质 平面PH曲线的构造及其相应性质PH曲线理论三次PH曲线的构造及性质四次PH曲线的构造及性质五次PH曲线的构造及性质非尖点五次PH曲线尖点五次PH曲线 参考文献 平面PH曲线的构造及其相应性质 过渡曲线常需要满足在连接点处位置连续、曲率连续以及切线…

如何看待 2023 OPPO 开发者大会?潘塔纳尔进展如何?AndesGPT 有哪些亮点?

在2023年11月16日举行的OPPO开发者大会&#xff08;ODC23&#xff09;上&#xff0c;OPPO带来了全新ColorOS 14、全新互联网服务生态以及健康服务进展&#xff0c;这些新动态中有许多值得关注的地方。 1、全新ColorOS 14&#xff1a; 效率提升&#xff1a;ColorOS 14通过一系列…

java基于springboot公益帮学网站 新闻发布系统的设计与实现vue

以Java为开发平台&#xff0c;综合利用Java Web开发技术、数据库技术等&#xff0c;开发出公益帮学网站。用户使用版块&#xff1a;可以选择注册并登录&#xff0c;可以浏览信息、可以网上互动、发布文章、内容推荐等。后台管理员管理版块&#xff1a;以管理员身份登录网站后台…