Java checked 异常 和 RuntimeException(运行时异常)

目录

   一、运行时异常
   
     1、什么是RuntimeExceptioin
     2、运行时异常的特点
     3、如何运用运行时异常
     
   二、运行时异常和ckecked异常的区别
   
     1、机制上
     2、逻辑上  
      

一、运行时异常

1、什么是运行时异常

程序在运行过程中出现的异常,RumtimeException 是 Exception 的一个子类
我们可以查看Java API 文档

贴张示意图
这里写图片描述

2、运行异常的特点

一般来说,我们在方法体内出现异常,我们用throw 关键字 将 异常对象或 异常对象的引用抛出,如果当前方法无法处理异常,那么必须在方法的参数列表后方法体前 必须 用 throws 声明异常所属类,交给调用者去处理。但是RuntimeException是非常特殊的子类,你可以不用throw和throws,哪怕你throw了,也没必要thtows,即使你throws了,调用者也没必要try-catch

1> 如果在函数内容中抛出该类异常或其子类异常,函数上可以不用声明,编译一样通过

2> 如果在函数上声明该异常,调用者可以不同处理(try-catch),编译一样通过

为什么不用声明 ?

之所以不用在函数上声明,是因为不需要让调用者处理,当该异常发生,希望程序停止,因为在运行时,出现了无法继续运算的情况,希望程序停止后由程序员对代码进行修正。

3、如何运用运行时异常

自定义异常时,如果该异常的发生,无法继续进行运行,就让自定义异常类继承RuntimeException

再贴一张简单示意图 Object 中的 wait(long timeout)方法,继续查看Java API 文档 对该方法的描述
这里写图片描述

二、运行时异常和ckecked异常的区别

对于异常分两种:检查异常和非检查异常(运行时异常)

检查异常 : 编译时被检测的异常 (throw后,方法有能力处理就try-catch处理,没能力处理就必须throws)。编译不通过,检查语法(其实就是throw和throws的配套使用)。

运行时异常 : 编译时不被检查的异常(运行时异常。RuntimeException及其子类)。编译通过。

1、机制上

主要表现在以下两个方面 :
(1)如何定义方法
(2)如何处理抛出的异常

运行时异常,不需要用throws 声明抛出 异常对象所属类,也可以不用throw 抛出异常对象或异常引用。对于调用该方法,也不需要放于 try-catch 代码块中。(为什么 ? 如果你捕获它,就会冒这么一个风险:程序代码错误被掩盖在运行中无法察觉)

而检查异常 : 一旦 用throw 抛出异常,如果当前方法 可处理异常,那么直接在该方法内用try-catch 去处理。如果当前方法不具备处理该异常的能力,那么就必须在 参数列表后方法体前用 throws 声明 异常 所属类,交给调用该方法的 调用者(方法) 去处理 。

2、逻辑上

从逻辑的角度来看, checked 异常 和 RuntimeException 有着不同的使用目的,检查性异常 用来指示 一种调用方能够直接处理的异常情况(例如: 用户输入错误,程序可以直接捕获并处理,提示用户输入错误), 而RuntimeException 是用来指 调用方 本身无法 处理或回复 的程序错误(例如,你封装个库给别人用,当别人调用你库中某个方法是,需要传入某些参数,如果用户传入的参数不合法,你自己没办法处理,那么刺客你抛出的就应该是运行时异常)。

我个人感觉,RuntimeException 跟OC 中的 断言(NSAlert) 有点像哦

进一步思考 : 前面的知识点有降到,携带参数的构造方法既可以为对象赋初始值,还可以为对象 赋一个合理的值。例如,初始化一个长方形对象,用构造方法给定 width 和 length ,我们可以在构造方法里面判断 长 宽 值的合理性,但是如果此时你初始化这个对象成功后,你调用 求面积的方法,返回一个 值,那么这个值 是异常值。按照所学的知识,我们一般会在求面积的方法,判断值的合理性,如果不合理抛出异常。但是,反过来想,如果我一开始让你传入 的 width 和 length 就抛出异常呢,这个直接让你初始化对象不成功 是不是好点。因为你有可能不知道 我 求面积的方法是怎么实现的,就算我给你抛出异常,你也不知道如何处理。(通过这点,我们也了解到一个小知识点,构造方法也是可以抛出异常的,而且,抛出异常的类型通常是运行时异常)

在这里插入图片描述

  • 26
    点赞
  • 70
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
异常概述 •异常处理已经成为衡量一门语言是否成熟的标准之一,目前的主流编程语言如C++、C#、Ruby、 Python等,大都提供了异常处理机制。增加了异常处理机制后的程序有更好的容错性,更加健壮。 传统错误处理的缺陷 •传统错误处理机制,主要如下两个缺点:   –无法穷举所有异常情况:因为人类知识的限制,异常情况总比可以考虑到的情况多,总有“漏网之鱼”的异常情况,所以程序总是不够健壮。   –错误处理代码和业务实现代码混杂:这种错误处理和业务实现混杂的代码严重影响程序的可读性,会增加程序维护的难度。 使用try...catch捕获异常 •执行try块里的业务逻辑代码时出现异常,系统自动生成一个异常对象,该异常对象被提交给Java运 行时环境,这个过程被称为抛出(throw)异常。 •Java运行时环境收到异常对象时,会寻找能处理该异常对象的catch块,如果找到合适的catch块并 把该异常对象交给该catch块处理,那这个过程被称为捕获(catch)异常;如果Java运行时环境找 不到捕获异常的catch块,则运行时环境终止,Java程序也将退出。 异常的捕捉流程 Java异常体系 访问异常信息 •如果程序需要在catch块中访问异常对象的相关信息,可以通过调用catch后异常形参的方法来获 得。当Java运行时决定调用某个catch块来处理该异常对象时,会将该异常对象赋给catch块后的异 常参数,程序就可以通过该参数来获得该异常的相关信息。 •所有异常对象都包含了如下几个常用方法:   –getMessage():返回该异常的详细描述字符串。   –printStackTrace():将该异常的跟踪栈信息输出到标准错误输出。   –printStackTrace(PrintStream s):将该异常的跟踪栈信息输出到指定输出流。   –getStackTrace():返回该异常的跟踪栈信息。 异常处理 复制代码 try { 需要检测的代码; } catch(异常类 变量) { 异常处理代码; } finally { 一定会执行的代码; } 复制代码 Finally代码块只有一种情况不会被执行。就是在之前执行了System.exit(0)。 Java 7提供的多异常捕捉 •在Java 7以前,每个catch块只能捕捉一个异常。从Java 7开始,一个catch块可以捕捉多个异常。 –catch(异常1 | 异常 2 | 异常3 ex) –{ –} •多个异常之间用竖线隔开。 •多异常捕捉时,异常变量之前有隐式final修饰。 本文原创作者:pipi-changing 本文原创出处:http://www.cnblogs.com/pipi-changing/ 使用finally回收资源 •程序在try块里打开了一些物理资源(例如数据库连接、网络连接和磁盘文件等),这些物理资源都 必须显式回收。 •为了保证一定能回收try块中打开的物理资源,异常处理机制提供了finally块。不管try块中的代码是 否出现异常,也不管哪一个catch块被执行,finally块总会被执行。 异常处理的嵌套 •异常处理流程代码可以放在任何能放可执行性代码的地方,因此完整的异常处理流程既可放在try块 里,也可放在catch块里,也可放在finally块里。 •异常处理嵌套的深度没有很明确的限制,但通常没有必要使用超过两层的嵌套异常处理,层次太深的 嵌套异常处理没有太大必要,而且导致程序可读性降低。 Java 7的自动关闭资源的try语句 –try( – // 此处声明的资源, 系统可以自动关闭它。 –) –{ – // –} •对于自动关闭资源的try语句, 可以没有catch和finally——try块可以孤独地存在。 •自动关闭资源的try语句,有两个注意点:   –只有放在try后面的圆括号里的资源才会被关闭。   –能被自动关闭的资源必须实现Closeable或AutoCloseable接口。 Checked异常与Runtime异常Java异常被分为两大类:Checked异常和Runtime异常运行时异常)。所有 RuntimeException类及其子类的实例被称为Runtime异常;不是RuntimeException类及其子类 的异常实例则被称为Checked异常Checked异常的处理 •当前方法明确知道如何处理该异常,程序应该使用try...catch块来捕获该异常,然后在对应的catch 块中修改该异常。 •当前方法不知道如何处理这种异常,应该在定义该方法时声明抛出该异常。 Runtime异常的处理 •Runtime异常则更加灵活,Runtime异常无需显式声明抛出。 •如果程序需要捕捉Runtime异常,也可以使用try...catch块来捕捉Runtime异常。 使用throws声明抛出异常 •throws声明抛出异常的思路是:当前方法不知道应该如何这种类型的异常,该异常应该由上一级调 用者处理,如果main方法也不知道应该如何处理这种类型的异常,也可以使用throws声明抛出异 常,该异常将交给JVM处理。JVM对异常的处理方法是:打印异常跟踪栈信息,并中止程序运行,这 就是前面程序在遇到异常后自动结束的原因。 •throws声明抛出只能在方法签名中使用,throws可以声明抛出多个异常类,多个异常类之间以逗 号隔开。throws声明抛出的语法格式如下   –throws ExceptionClass1 , ExceptionClass2... 抛出异常 •如果需要在程序中自行抛出异常,应使用throw语句,throw语句可以单独使用,throw语句抛出 的不是异常类,而是一个异常实例,而且每次只能抛出一个异常实例。throw语句的语法格式如下:   –throw ExceptionInstance; •如果throw语句抛出的异常Checked异常,则该throw语句要么处于try块里,显式捕获该异常 ,要么放在一个带throws声明抛出的方法中,即把该异常交给该方法的调用者处理。 Java 7增强的throw语句 –try –{ – new FileInputStream(“a.txt”); –} –Catch(Exception ex) –{ – ex.printStackTrace(); – throw ex; //① –} •从JDK 7开始,Java编译器可以只能地识别①号代码处抛出的异常只是FileNotFoundException异常。 自定义异常类 •程序很少会自行抛出系统异常,因为异常的类名通常包含了该异常的有用信息。所以在选择抛出什么 异常时,应该选择合适的异常类,从而可以明确地描述该异常情况。在这种情形下,应用程序常常需要 抛出自定义异常。 •用户自定义异常都应该继承Exception基类,如果希望自定义Runtime异常,则应该继承 RuntimeException基类。定义异常类时通常需要提供两种构造器:一个是无参数的构造器;另一个 是带一个字符串参数的构造器,这个字符串将作为该异常对象的详细说明(也就是异常对象的 getMessage方法的返回值)。 异常链 •当业务逻辑层访问持久层出现SQLException异常时,程序不应该把底层的SQLException异常传 到用户界面,原因有如下两个:   –对于正常用户而言,他们不想看到底层SQLException,SQLException对他们使用该系统没 有任何帮助。   –对于恶意用户而言,将SQLException暴露出来是一种不安全的。 Java异常跟踪栈 •异常对象的printStackTrace方法用于打印异常的跟踪栈信息,根据printStackTrace方法的输出 结果,我们可以找到异常的源头,并跟踪到异常一路触发的过程。 •面向对象的应用程序运行时,经常会发生一系列方法调用,从而形成“方法调用栈”,异常的传播则与 相反:只要异常没有被完全捕获(包括异常没有被捕获,或异常被处理后重新抛出了新异常),异常从 发生异常的方法逐渐向外传播,首先传给该方法的调用者,该方法调用者再次传给其调用者……直至最 后传到 main方法,如果main方法依然没有处理该异常,JVM会中止该程序,并打印异常的跟踪栈信 息。 异常处理规则 •不要过度使用异常 •不要使用过于庞大的try块 •避免使用Catch All语句 •不要忽略捕获到异常 。。。。。。。。。。。。。
模块7 异常处理 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第1页。 学习目标 01 了解异常产生的原因。 02 了解Java异常体系结构,以及受检异常和未受检异常的区别。 03 掌握使用try-catch-finally语句块捕捉并处理异常。 05 掌握设计和使用用户自定义异常。 04 掌握使用throw语句抛出异常,以及使用throws语句声明方法异常。 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第2页。 技能目标 能够在MyEclipse IDE中使用try-catch-finally语句块编写程序捕捉并处理异常。 01 能够在MyEclipse IDE中使用throw语句编写程序抛出异常。 02 能够在MyEclipse IDE中使用throws语句编写程序声明方法异常。 03 能够在MyEclipse IDE中编写程序定义自己的异常。 04 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第3页。 7.1 回顾与思考 Java语言的健壮性确保了在编程过程中出现的错误可以得到及时、有效的控制。但是,并不是所有错误都能在编译阶段被发现。有些错误只有在运行时才会发生。在编译或运行时发生的错误都被称为异常。在编译或运行时跟踪这些异常被称为异常处理。Java异常处理机制负责异常处理和错误恢复。 如果把【例6-1】中的代码"for(int i = 1;i < myArray.length;i++)"改成"for(int i = 1;i <= myArray.length;i++)",这时,编译可以通过。运行后在控制台出现如下提示。 Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 8 at ch6.example.Example6_1.main(Example6_1.java:16) 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第4页。 7.2 异常异常类 7.2.1 Java异常体系结构 Java中的异常有很多类型,包括在Java API中已经被定义的异常(也称系统异常)和用户自定义异常。所有的异常类都直接或间接继承了Throwable类。Throwable类是Object类的直接子类。 Throwable类有两个直接子类: Error类 Exception类 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第5页。 7.2 异常异常类 7.2.2 受检异常 Exception类及其子类中除了RuntimeException类及其子类以外的异常被称为受检异常checked exception)。受检异常在编译阶段能被Java编译器检测到,不可回避,用户必须对受检异常进行处理。 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第6页。 7.2 异常异常类 7.2.2 受检异常 【例7-1】受检异常示例。 文件名为Example7_1.java,其代码如下。 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第7页。 7.2 异常异常类 7.2.2 受检异常 【例7-1】受检异常示例。 在"Class.forName("com.mysql.jdbc.Driver");"这条语句下出现了波浪线,光标停留在该语句上会出现浮窗提示:未处理异常类型ClassNotFoundException,如图7-1所示。 图7-1 受检异常未处理的浮窗提示 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第8页。 7.2 异常异常类 7.2.3 未受检异常 属于RuntimeException类及其子类的异常和Error类及其子类的异常被称为未受检异常(unchecked exception)。未受检异常在编译阶段不能被Java编译器检测到。不强迫用户利用异常处理机制对未受检异常进行处理。未受检异常的发生是因为程序有逻辑错误。未受检异常首先且主要通过用户在程序编写过程中避免。如果滥用异常处理机制处理(预防)未受检异常,程序会变得臃肿,影响可读性。 《Java程序设计案例教程》教学课件07异常处理全文共31页,当前为第9页。 7.3 异常处理结构 7.3.1 捕捉异常 1. try-catch语句块 可以使用try和catch语句块对程序中可能出现的异常进行捕捉并处理。把可能产生异常的代码放置在try语句块内,catch语句块紧接着try语句块之后。catch语句块中包含了处理异常及输出异常产生原因的语句。一个catch语句块仅对应一个try语句块,但是一个try语句块可以对应多

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值