【Java挠头】Java异常、捕获、处理、throw、throws等绝妙剖析

简介: 【Java挠头】Java异常、捕获、处理、throw、throws等绝妙剖析

–本文摘要–

   1、本文的目标读者:是接触过Java异常,但是仅限于书本和一些例题,某一天写着写着会突然觉得“‘异常’没什么意义,Java不是已经帮我们处理了嘛?”的这些朋友。

   2、本文的内容摘要:讲述日常开发中如何看待‘异常’这个概念,讲述在一个中型web项目中,我们如何处理‘异常’。

   【版权声明】归CSDN账户[征途黯然.]/公众号[三黄工作室]原创,禁止任何网站与个人采集或转载。

一、本文结构


image.png

  读完本文,你一定能解开如下困惑:


1)Java自己会报错,而且很多网站(比如xx大学的官网)路径错误时,就会报一堆Java错误,网站中这些错误给用户的提示,都在报Java原始的错误信息,那还要学异常处理的意义何在?

2)throw和throws有何区别?在方法里面throw错误不是有病嘛?throws在方法名那一行定义到底完成了哪些功能?

3)一个正规项目中,异常多种多样,开发者是不是要写很多异常处理?

4)夏天到了,小姐姐们会不会因为没有男朋友而产生exception,要不要我去处理一下?【哈哈】

二、大学官网让我觉得‘异常处理毫无意义’


   💻言归正传。

   当年刚学Java的异常,觉得Java真厉害啊,异常先捕获下来,分分钟处理掉。结果,一登学校官网,时不时给我报个变量‘xxx’没有赋值。当时我就在想,如果这种基础错误开发者都不做处理,那还要Java中还要‘捕获’这个概念干嘛?大家一起用Java自带的一大串异常报错不就好了。

三、Java异常


   基础的Java异常关系如下图3-1所示:

20210514200302423.png

  【首先】

   Exception是程序执行时发生的异常,Error是JVM运行错误(这是致命错误),无法处理。他们共有一个基类是Throwable,所以在日常开发中,我们对Exception写的多一点,因为Error是致命错误。

   异常可以理解为‘程序运行中一个小毛病’。发生Error,程序都跑不起来,何来小毛病一说呢。


   【其次】

   这里把RuntimeException贴出来的目的是,我们项目中用到的异常类型,我们一般都定义为RuntimeException类型。用RuntimeException的原因,在文章最后有介绍。

四、捕获与处理


   异常,是程序中导致程序中断的指令流。如果我们不去管它,那么发生了异常,程序就会中断,这是在日常项目中绝对不能发生的(得亏多少钱)。所以,我们要处理它,用一种合适的方式处理,所以用到了try-catch-finally语法。见下图4-1:

20210514202818779.png


   【图4-1解释】try-catch-finally语法就像是一个容器,程序执行时,代码C可能发生异常而导致程序中断。我们加一个容器来包含住代码C,这样无论它出错与否,起码不会让程序中断,连个return都不能执行。

  关于try-catch-finally初学者会产生很多疑问。⭐⭐比较突出的是:既然代码C出错了,如果代码D需要用到代码C里面的变量,我继续执行代码D有什么意义呢?

   这里引出我们在开发中使用异常捕获的目的,见下图4-2:

20210514213931140.png

  【捕获异常的意义】

   首先要保证捕获内容的原子性(抽象上的原子性),也就是try的内容尽量独立,不要被try外部引用,否则捕获是不完整的。


   【技术上的意义】

1、捕获异常,防止方法直接中断,否则可能连return都无法执行。

2、处理可能出错的变量。如果是该变量异常,我们在try之后,可以给变量设默认值,保证它在接下来可以良好运行。

3、回收空间,保证系统性能。如果一块空间出错,我们可以回收,防止性能消耗。


   【业务上的意义】

   我们在出现异常后,可以设计一个监听器来识别系统中所有的异常错误,对不同类型的异常,返回不同的信息给用户,这样可以保证用户看得懂(系统的友好性)。

五、throw与throws


   【throw】

   throw是在方法内部抛出一个异常(可以自定义,也可以是Java自带的异常类)。手动抛出的异常,只要类型相同,和系统自己产生的异常没差别,请看以下代码:

  @Test
    public void test001(){
        int a=2/0;
    }
    @Test
    public void test003(){
        throw new ArithmeticException("不可为零");
    }

  代码解释:int a=2/0;在Java中会报ArithmeticException错误,与手动throw一个ArithmeticException错误是等效的。两个测试函数的报错大致相同,见下图5-1:

2021051422072528.png

   关于throw初学者还是会产生很多疑问。⭐⭐大概是:为什么要在程序中自己抛出异常?不是应该避免程序中的异常嘛?

   这个问题的解答请看本文的最后一节。

   【throws】

   throws是在方法定义时,添加异常声明。throws可以声明多个异常类,声明的基本语法如下代码:

  @Test
    public void test001() throws  Exception,MyException{
        int a=2/0;
    }

   throws声明了本方法中可能出现的异常类。表示如果方法中真的产生了异常,Java程序不会立即中断,而是会把异常向上传递给上一级,让上一级来处理,而上层的方法必须处理这个异常,不处理会出现Error。

 而普通定义的方法,上层方法不是一定要定义try-catch机制。

    处理逻辑见下图5-2:

20210514231738491.png

六、日常开发中的异常处理


20210514235812151.png

⭐【第三节遗留问题】项目开发中,用RuntimeException的原因。


   【–解答–】RuntimeException类型的异常,支持开发者自定义处理,也支持开发者不处理,让Java来兜底。这样自由度高,开发简易,开发中我们大多使用RuntimeException类型来派生出自定义异常类型。


    ⭐【第五节遗留问题】为什么要在程序中自己抛出异常?不是应该避免程序中的异常嘛?


   【–解答–】我们在方法中抛出异常,提交给项目中的异常监听器来处理异常,分析异常中的类型,然后给用户返回对应的提示信息。

相关文章
|
9月前
|
Java 程序员 数据库连接
我们详细地讲解一下 Java 异常及要如何处理
我是小假 期待与你的下一次相遇 ~
205 1
|
SQL druid Oracle
【YashanDB知识库】yasdb jdbc驱动集成druid连接池,业务(java)日志中有token IDENTIFIER start异常
客户Java日志中出现异常,影响Druid的merge SQL功能(将SQL字面量替换为绑定变量以统计性能),但不影响正常业务流程。原因是Druid在merge SQL时传入null作为dbType,导致无法解析递归查询中的`start`关键字。
|
SQL Java 中间件
【YashanDB知识库】yasdb jdbc驱动集成BeetISQL中间件,业务(java)报autoAssignKey failure异常
在BeetISQL 2.13.8版本中,客户使用batch insert向yashandb表插入数据并尝试获取自动生成的sequence id时,出现类型转换异常。原因是beetlsql在prepareStatement时未指定返回列,导致yashan JDBC驱动返回rowid(字符串),与Java Bean中的数字类型tid不匹配。此问题影响业务流程,使无法正确获取sequence id。解决方法包括:1) 在batchInsert时不返回自动生成的sequence id;2) 升级至BeetISQL 3,其已修正该问题。
【YashanDB知识库】yasdb jdbc驱动集成BeetISQL中间件,业务(java)报autoAssignKey failure异常
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
743 14
|
缓存 Java 应用服务中间件
java语言后台管理若依框架-登录提示404-接口异常-系统接口404异常如何处理-登录验证码不显示prod-api/captchaImage 404 (Not Found) 如何处理-解决方案优雅草卓伊凡
java语言后台管理若依框架-登录提示404-接口异常-系统接口404异常如何处理-登录验证码不显示prod-api/captchaImage 404 (Not Found) 如何处理-解决方案优雅草卓伊凡
3513 5
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
285 1
|
5月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
307 1
|
6月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
269 0
|
6月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
456 16