【Java零基础入门篇】第 ⑥ 期 - 异常处理

简介: Java零基础入门

博主:命运之光
专栏:JAVA入门

JAVA.gif

学习目标

掌握异常的概念,Java中的常见异常类;

掌握Java中如何捕获和处理异常;

掌握自定义异常类及其使用;


异常概述

程序员在写代码的时候即便写的尽善尽美,在系统的运行过程中依据会遇到一些问题,因为很多问题不是靠代码能够避免的。

异常体系

🍓🍓异常是处理运行时错误的特殊类,所有异常类都是java.lang.Throwable的子类,Throwable 类是类库java.Lang 包中的一个类,它派生了两个子类:ExceptionError

  • Error:Java虚拟机无法解决的严重问题。如:JVM系统内部错误、资源耗尽等严重情况。必须修改代码,程序才可以继续执行。
  • Exception其它因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。例如:空指针访问、试图读取不存在的文件、网络连接中断、数组下标越界等。

Exception分为:运行时异常(非受检异常)编译时异常(受检异常)

编译时异常:类型上都属于Exception类及其子类。编译时为了保证程序的健壮性 ,可以提前预料到的异常,从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过

运行时异常:都是RuntimeException类及其子类异常,如NullPointerException(空指针异常)、IndexOutOfBoundsException(下标越界异常)等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。

运行时异常的特点是Java编译器不会检查它,也就是说,当程序中可能出现这类异常,也会编译通过。

常见的异常

1.ArithmeticException 异常

算术运算异常,当除数为零时就会产生此类异常。

例如:int  x=2/0;              //此时会发生ArithmeticException异常

2.ArrayIndexOutOfBoundsException异常

访问数组元素的下标越界异常,当访问的数组元素的下标超过数组的上下界时发生此类异常。

例如:   int a[]=new int[3]; //定义了数组a,下标从0到2是合法的 int x=a[3]; //下标超过上界,此时会发生ArrayIndexOutOfBoundsException异常

3.ArrayStoreException异常

数组存储异常,当数组存储空间不够或类型不匹配时会产生此类异常。

例如:   int a[]=new int[5];   String b=new String[5];   String.arraryCopy(a,b);       //类型不匹配会产生ArrayStoreException异常

4. NegativeArraySizeException异常

负值数组大小异常,当数组的存储单元个数为负数时会产生此类异常。例如:   int a[]=new int[-5];                     //数组个数大小为负数,发生NegativeArraySizeException异常。

5.IndexOutOfBoundsException异常

索引(下标)越界异常,是ArrayIndexOutOfBoundsException的父类。例如:   String s=”abc123”;   s.charAt(8);      // 下标8越界,发生IndexOutOfBoundsException异常

6.NullPointerException异常

访问空引用(指针)异常。例如:   String s=null;   s.charAt(2);                    //s为null,发生NullPointerException异常

7.  NumberFormatException异常

数值格式异常,当数据类型格式转换不匹配时会产生此类异常。

8.  ClassCaseException异常

类强制转换异常,当把一个类强制转换成另一个没有任何继承关系的类时产生此类异常。

9. StringIndexOutOfBoundsException异常

字符串下标越界异常。

10. SecurityException异常

安全性异常。

Java的异常处理机制

🍓🍓在编写程序时,经常要在可能出现错误的地方加上检测的代码,如进行x/y运算时,要检测分母为0,数据为空,输入的不是数据而是字符等。

🍓🍓Java中针对各种运行错误定义了很多异常类每个异常类都代表了一种运行错误。每当java程序运行过程中某段代码发生一个可识别的运行错误时,JVM都会产生一个相应的异常类的对象对象封装了异常的有关信息

Java提供的异常处理机制是采取抛出异常,然后捕获异常的处理形式,主要分为两个步骤:

  • 抛出异常:在程序运行中当语义规则被违反时,在异常代码处即产生一个异常事件,生成一个对应异常类的对象,将此对象抛出(throw) ,其后的代码不再处理。
  • 捕获异常:如果一个方法内抛出异常,该异常对象会被抛给调用者方法中处理。如果异常没有在调用者方法中处理,它继续被抛给这个调用方法的上层方法。这个过程将一直继续下去,直到异常被处理。这一过程称为捕获(catch)异常

如果一个异常回到main()方法,并且main()也不处理,则程序运行终止;

程序员通常只能处理Exception,而对Error无能为力.

方式一:try – catch - finally

语法格式:

try {   
// 有可能出现异常的语句} catch (异常类型1对象) {  
// 异常处理 ;} catch (异常类型2对象) { 
// 异常处理 ;} catch (异常类型3对象) {  
// 异常处理 ;} .... 
finally {     
//不管是否出现异常,都执行的统一代码}

🍓🍓例1:直接使用对应异常类作为参数(try – catch - finally)。

publicclasstext1 {
publicstaticvoidmain(Stringargs[]) {
System.out.println("1、除法计算开始。");
try {
System.out.println("2、除法计算:"+ (10/0));   // 此处产生异常// 异常产生之后的语句将不再执行,此处在try中产生异常,所以下面的输出不会执行System.out.println("异常产生后的语句");
        } catch (ArithmeticExceptione) {  // 处理算术异常System.out.println("******** 出现异常了 *********");
        }
System.out.println("3、除法计算结束。");
    }
}

说明:

  • try: 捕获异常的第一步是用try{…}语句块选定捕获异常的范围,将可能出现异常的代码 放在try语句块中,一旦出现异常,就会生成一个对应异常类的对象;
  • catch (Exceptiontype e): catch语句块中是对异常对象进行处理的代码。每个try 语句块可以伴随一个或多个catch语句,用于处理可能产生的不同类型的异常对象。
  • 如果明确知道产生的是何种异常,可以用该异常类作为catch的参数;也可以用其父类作为catch的参数。
  • 比 如 : 可以用 ArithmeticException 类(子类)作为参数的地方,就可以用 RuntimeException类(父类)作为参数,或者用所有异常的父类Exception类作为参数。

🍓🍓例2:使用异常类对应的父类作为参数(try – catch - finally)。

publicclasstext1 {
publicstaticvoidmain(Stringargs[]) {
System.out.println("1、除法计算开始。");
try {
System.out.println("2、除法计算:"+ (10/0));   // 此处产生异常// 异常产生之后的语句将不再执行,此处在try中产生异常,所以下面的输出不会执行System.out.println("异常产生后的语句");
        } catch (RuntimeExceptione) {  // 处理算术异常System.out.println("******** 出现异常了 *********");
        }
System.out.println("3、除法计算结束。");
    }
}

注意:

  • 一旦try中的异常对象匹配到某一个catch时,进入catch中进行异常的处理,一旦处理完成,跳出try-catch结构(没写finally的情况),继续执行其它代码;
  • finally: 捕获异常的最后一步是通过finally语句为异常处理提供一个统一的出口,使得在控制流转到程序的其它部分以前,能够对程序的状态作统一的管理。
  • 不论在try代码块中是否发生了异常事件,catch语句是否执行,catch语句是否有异常,catch语句中是否有return,finally块中的语句都会被执行。
  • finally语句和catch语句是任选的。
  • 🍓catch中的异常类型如果没有子父类关系,则声明顺序没有要求,谁在上均可。
  • 🍓catch中的异常类型如果有子父类关系,则子类声明必须在父类声明之上否则报错

捕获异常的有关信息:    

与其它对象一样,可以访问一个异常对象的成员变量或调用它的方法

  • getMessage() :获取异常说明信息,返回字符串
  • printStackTrace() :输出异常类名和异常信息,以及异常出现在程序中的位置。返回值void

🍓🍓例3:输出异常完整信息。

publicclasstext1 {
publicstaticvoidmain(Stringargs[]) {
System.out.println("1、除法计算开始。");
try {
System.out.println("2、除法计算:"+ (10/0));// 此处产生异常// 异常产生之后的语句将不再执行,此处在try中产生异常,所以下面的输出不会执行System.out.println("若前面代码出现异常,此行代码不会执行!");
        } catch (ArithmeticExceptione) {   // 处理算术异常e.printStackTrace();    // 输出异常的完整信息        } finally {
System.out.println("### 不管是否出现异常我都执行!") ;
        }
System.out.println("3、除法计算结束。");
    }
}

方式二:throws+异常类型

🍓🍓Java中允许在方法的后面使用throws关键字对外声明该方法有可能发生的异常,但此方法中不处理异常。

🍓🍓这样调用者在调用方法时,就明确地知道该方法可能有异常,并且必须在程序中对异常进行处理,否则编译无法通过。

throws关键字声明抛出异常的语法格式如下:

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

🍓🍓例:class  MyMath

声明如下方法可能会抛出一个ArithmeticExcption或IOException类型的异常,使它的调用者知道要捕获这个异常。

classMyMath { 
publicstaticintdiv(intx, inty) throwsArithmeticException, IOException {             
// 此方法不处理异常     returnx/y;   
}
}

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

🍓主方法也可以使用throws抛出异常,这时主方法里可以不用强制进行异常处理,而是将异常处理交给JVM进行默认处理,此时会导致程序中断。(不建议使用

手动抛出异常

🍓🍓Java异常类对象除在程序执行过程中出现异常时由系统自动生成并抛出,也可根据需要使用人工创建并抛出

手动创建并抛出异常,通过throw关键字。

throw语句用来直接抛出一个异常,后接一个可抛出的异常类对象

格式如下:

thrownew异常类名(【参数列表】)

异常类名对象名=new异常类名(【参数列表】)throw对象名;


例如:首先要生成异常类对象,然后通过throw语句实现抛出操作(提交给Java运行环境)

IOExceptione=newIOException( ); 
throwe;

或:  

thrownewIOException();

说明:

  • 可以抛出的异常必须是Throwable或其子类的实例。下面的语句在编译时将会产生语法错误:throw new String("want to throw");

重写方法声明抛出异常的原则

  • 一个方法被覆盖时,覆盖它的方法必须抛出跟被覆盖的方法相同的异常或者异常的子类,或者不抛出异常;
  • 如果父类抛出多个异常,那么重写(覆盖)的方法必须抛出那些异常的一个子集,也就是说不能抛出新的异常。

throws和throw的区别:

  • 首先throws出现在方法函数头(声明处),表示若抛出异常,由该方法的调用者来处理;
  • throw出现在函数体,表示在方法中手工抛出一个异常。

自定义异常

🍓🍓创建用户定义的异常类,可通过扩充(继承)Exception类创建自己的异常类,被扩充的异常类继承Exception类的属性和方法,具有异常类全部特性,同时也和任何其他类一样,可以有数据成员、构造方法及其他功能性的成员方法

自定义异常类后,创建异常类对象,最后再使用throw语句抛出该异常对象,在调用点添加异常处理语句即可。

自定义异常步骤

第1步:自定义异常类。例如:

classMyExceptionextendsException {   
……}

第2步:定义异常对象,并抛出该对象。

式1:

thrownewMyException ( );

方式2:

MyExceptione=newMyException( );       
throwe;

相关文章
|
1月前
|
Java 开发者
Java中的异常处理:从基础到高级
在Java编程的世界里,异常处理是一块基石,它确保了程序的健壮性和稳定性。本文将带你从异常的基础概念出发,逐步深入到高级处理技巧,通过实例展示如何在Java中有效管理和处理异常。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
1月前
|
Java 程序员
Java编程中的异常处理:从基础到高级
在Java的世界中,异常处理是代码健壮性的守护神。本文将带你从异常的基本概念出发,逐步深入到高级用法,探索如何优雅地处理程序中的错误和异常情况。通过实际案例,我们将一起学习如何编写更可靠、更易于维护的Java代码。准备好了吗?让我们一起踏上这段旅程,解锁Java异常处理的秘密!
|
3天前
|
自然语言处理 Java
Java中的字符集编码入门-增补字符(转载)
本文探讨Java对Unicode的支持及其发展历程。文章详细解析了Unicode字符集的结构,包括基本多语言面(BMP)和增补字符的表示方法,以及UTF-16编码中surrogate pair的使用。同时介绍了代码点和代码单元的概念,并解释了UTF-8的编码规则及其兼容性。
75 60
|
1月前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
58 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
29天前
|
监控 架构师 Java
Java虚拟机调优的艺术:从入门到精通####
本文作为一篇深入浅出的技术指南,旨在为Java开发者揭示JVM调优的神秘面纱,通过剖析其背后的原理、分享实战经验与最佳实践,引领读者踏上从调优新手到高手的进阶之路。不同于传统的摘要概述,本文将以一场虚拟的对话形式,模拟一位经验丰富的架构师向初学者传授JVM调优的心法,激发学习兴趣,同时概括性地介绍文章将探讨的核心议题——性能监控、垃圾回收优化、内存管理及常见问题解决策略。 ####
|
29天前
|
Java
Java 异常处理:11 个异常处理最佳实践
本文深入探讨了Java异常处理的最佳实践,包括早抛出晚捕获、只捕获可处理异常、不忽略异常、抛出具体异常、正确包装异常、记录或抛出异常但不同时执行、不在finally中抛出异常、避免用异常控制流程、使用模板方法减少重复代码、抛出与方法相关的异常及异常处理后清理资源等内容,旨在提升代码质量和可维护性。
|
1月前
|
安全 Java 数据库连接
Java中的异常处理:理解与实践
在Java的世界里,异常处理是维护代码健壮性的守门人。本文将带你深入理解Java的异常机制,通过直观的例子展示如何优雅地处理错误和异常。我们将从基本的try-catch结构出发,探索更复杂的finally块、自定义异常类以及throw关键字的使用。文章旨在通过深入浅出的方式,帮助你构建一个更加稳定和可靠的应用程序。
31 5
|
30天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
123 1
|
1月前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
51 2
|
1月前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
51 2