除了自定义类,java中还为我们提供了一系列的工具类。我们会为大家介绍6种最常用的工具类。
- 如何进行应用异常处理程序中的问题?(系统介绍) - 如何通过包装器类实现基本数据类型的对象化处理 - String、StringBuilder是如何进行字符串信息操作的 - 常用集合框架及实现类使用 - 如何使用Java输入输出流进行文件读写 - 如何使用多线程实现数据并发通信(难度从上向下依次递增)
异常课程介绍
什么是异常?如何处理异常?
try-catch-finally;throw;throws;自定义异常;异常链
什么是异常?
异常: 意外、例外。异常本质上是程序上的错误。
错误在我们编写程序的过程中会经常发生,包括编译期间和运行期间的错误。
括号没有正常的配对; 语句结束后少写了分号;关键字编写错误 前面这几种都是编译期间的错误。通常这种错误编译器会帮助我们一起进行修订
但是运行期间的错误,编译器就无能为力了。
前面我们遇到的程序中的异常:
使用空的对象引用调用方法
String str=null; System.out.println(str.length());
数组访问时下标越界
int[] ary={1,2,3};for(int i=0;i<=3;i++){ System.out.println(ary[i]); }
算术运算时除数为0
int one =12;int two=0; System.out.println(one/two);
类型转换时无法正常转型
class Animal { }class Dog extends Animal { }class Cat extends Animal{ }public class Test { public static void main(String[] args) { Animal a1 = new Dog(); Animal a2 = new Cat(); Dog d1 = (Dog)a1; Dog d2 = (Dog)a2; }
以上都是运行期间的错误。以上这些代码在编译时是没有错误提示的。
在程序运行过程中,意外发生的情况,背离我们程序本身的意图的表现,都可以理解为异常。
错误执行 & 执行到崩溃
如何针对程序运行期间产生的异常进行合理的处理?
Java中有强大的异常处理机制。
异常分类
异常可以理解为一种事件,当它发生时会影响正常的程序运行流程。
Java中通过Throwable来进行各种异常信息的描述。
Error是程序无法处理的错误,表示运行应用程序中较严重问题。Java虚拟机问题(虚拟机错误 VirtualMachineError;内存溢出OutOfMemoryError;线程死锁 ThreadDeath);它们在应用程序的控制和处理能力之外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。
Exception是程序本身可以处理的异常。异常处理通常指针对这种类型异常的处理。
Exception可以分为两大类: 非检查异常(Unchecked Exception); 检查异常(Checked Exception)
非检查异常是编译器不要求强制处置的异常,它包括RuntimeException以及它的子类: 空指针异常 NullPointerException;数组下标越界异常 ArrayIndexOutOfBoundsException;算数异常 ArithmeticException;类型转换异常 ClassCastException;
java程序在编译阶段是不会去检查上面这些runtime异常的
检查异常,编译器要求必须处置的异常。如: IO异常 IOException; SQL异常 SQLException
异常处理分类
在Java应用程序中,异常处理机制为: 抛出异常;捕捉异常
异常要先被抛出,然后才能被捕获。
抛出异常指的就是当一个方法当中出现异常,方法会去创建异常对象,并去交付给运行时系统来进行处理。
异常对象: 异常类型以及异常出现时的程序状态等
当运行时系统捕获到这个异常,此时会去寻找合适的处理器,如果找到了,就会执行处理器的相关逻辑;如果始终没找到,运行就会终止
对于运行时异常、错误或可查异常, Java技术所要求的异常处理方式有所不同。
Java规定: 对于可查异常(检查异常(Checked Exception))必须捕捉、或者声明抛出
允许忽略不可查的RuntimeException(含子类)和Error(含子类)。
对于抛出异常和捕获异常,Java中通过5个关键字来实现: try catch finally throw throws
其中try catch finally是一组,是用来捕获异常的。try 执行可能产生异常的代码; catch 捕获异常;finally 无论是否发生异常代码总能执行;
throws 声明可能要抛出的异常; throw 手动抛出异常
try-catch-finally简介
public void method(){ try { //代码段1 //产生异常的代码段2 }catch (异常类型 ex) { //对异常进行处理的代码段3 }finally{ //代码段4 } }
try块后可接零个或多个catch快,如果没有catch块则必须跟一个finally块。简单来讲就是try要和catch或finally组合使用,不可单独存在。catch和finally如果没有try的加入,也是无法起作用的。
package cn.mtianyan.exception;import java.util.Scanner;public class TryDemo { public static void main(String[] args) {// // 定义两个整数,输出两数之商// int one = 12;// int two = 2;// System.out.println("one/two="+one/two); // 用户输入不可控 System.out.println("====运算开始===="); Scanner scanner = new Scanner(System.in); System.out.print("请输入第一个数字: "); int one = scanner.nextInt(); System.out.print("请输入第二个数字: "); int two = scanner.nextInt(); System.out.println("one/two="+one/two); System.out.println("====运算结束===="); } }
注释部分是我们在程序写死的数字逻辑,是可以正常运行的。但一旦我们将这些变量交给了用户,那么用户输入的不可控因素就会给程序带来异常。
Exception in thread "main" java.lang.ArithmeticException: / by zero at cn.mtianyan.exception.TryDemo.main(TryDemo.java:19)
Exception in thread "main" java.util.InputMismatchException at java.base/java.util.Scanner.throwFor(Scanner.java:939) at java.base/java.util.Scanner.next(Scanner.java:1594) at java.base/java.util.Scanner.nextInt(Scanner.java:2258) at java.base/java.util.Scanner.nextInt(Scanner.java:2212) at cn.mtianyan.exception.TryDemo.main(TryDemo.java:16)
try-catch结构进行异常处理
将可能出现异常的代码放在try块里。
package cn.mtianyan.exception;import java.util.Scanner;public class TryDemo { public static void main(String[] args) { // 用户输入不可控 System.out.println("====运算开始===="); try { Scanner scanner = new Scanner(System.in); System.out.print("请输入第一个数字: "); int one = scanner.nextInt(); System.out.print("请输入第二个数字: "); int two = scanner.nextInt(); System.out.println("one/two=" + one / two); }catch (Exception e){ System.out.println("程序出错啦"); } System.out.println("====运算结束===="); } }
有没有办法可以提示我是哪里出错,出的什么错呢?
catch (Exception e){ System.out.println("程序出错啦"); e.printStackTrace(); }
printStackTrace输出的格式有些诡异。错误栈应该是倒着看,最后一行是我们自己代码中的位置。
finally { System.out.println("====运算结束===="); }
finally块保证一定会被执行。
使用多重catch结构处理异常
针对不同的异常有不同的处理方式该如何做到?
catch (ArithmeticException e){ System.out.println("除数不可以为0"); e.printStackTrace(); }catch (InputMismatchException e){ System.out.println("请输入整数"); e.printStackTrace(); }catch (Exception e){ System.out.println("程序出错了"); }
推荐最后使用Exception兜底,必须放在最后一个。
package cn.mtianyan.exception;public class TestExercise { public static void main(String args[]) { try { int a = 1 - 1; System.out.println("a = " + a); int b = 4 / a; int c[] = {1}; c[10] = 99; } catch (ArithmeticException e) { System.out.println("除数不允许为0"); } catch (ArrayIndexOutOfBoundsException e) { System.out.println("数组越界"); } } }
上述代码的运行结果为:
第一次发生异常被捕获之后,程序去执行了catch块中的语句,不会再去执行首次出现异常行之下的try中代码
终止finally执行方法
通常情况,finally块是一定会执行的。
catch (ArithmeticException e){ System.exit(1); System.out.println("除数不可以为0"); e.printStackTrace(); }
System.exit(1);终止当前虚拟机运行。程序无条件终止运行。
0代表正常退出,非0代表非正常退出。如果程序按照正常逻辑执行需要退出时,调用System.exit(O)。如果发生异常而退出,比如在catch块使用,则调用System.exit(1);
return关键字在异常处理中的作用
之前的学习中我们知道return关键字可以用于方法返回值的带回。
package cn.mtianyan.exception;import java.util.Scanner;public class TryDemoReturn { public static void main(String[] args) { System.out.println("one/two="+add()); } public static int add(){ // 用户输入不可控 System.out.println("====运算开始===="); try { Scanner scanner = new Scanner(System.in); System.out.print("请输入第一个数字: "); int one = scanner.nextInt(); System.out.print("请输入第二个数字: "); int two = scanner.nextInt(); return one/two; }catch (ArithmeticException e){ System.out.println("除数不可以为0"); return 0; } finally { System.out.println("====运算结束===="); return -9999; } } }
运行结果:
可以看到,不管有多少个return在前面,都会以最后一个finally中的return为准作为返回值。
删除掉finally中的return语句,程序就可以按照我们预期的状态运行。
package cn.mtianyan.exception;public class TestExerciseTwo{ public static int test(int b){ try{ b+=10; return b; }catch(Exception e){ return 1; }finally{ b+=10; return b; } } public static void main(String[] args) { int num =10; System.out.println(test(num)); } }
运行结果:
使用throws声明异常类型
可以通过throws声明将要抛出何种类型的异常,通过throw将产生的异常抛出。
如果一个方法可能会出现异常,但没有能力处理这种异常,可以在方法声明处用throws子句来声明抛出异常。谁调用了这个方法,谁就该去处理这样的异常。
throws语句用在方法定义时声明该方法要抛出的异常类型。
public void method() throws Exception1,Exception2,...,ExceptionN { //可能产生异常的代码}
当方法抛出异常列表中的异常时,方法将不对这些类型及其子类类型的异常作处理,而抛向调用该方法的方法,由他去处理。
public static int add() throws ArithmeticException { // 用户输入不可控 System.out.println("====运算开始===="); Scanner scanner = new Scanner(System.in); System.out.print("请输入第一个数字: "); int one = scanner.nextInt(); System.out.print("请输入第二个数字: "); int two = scanner.nextInt(); System.out.println("====运算结束===="); return one / two; }
上面的代码在方法中声明可能抛出的异常,去除掉方法自己的异常处理,交给上层处理。可以采取快捷键来生成try catch包裹(生成时会自动检查异常类型)。
try { System.out.println("one/two=" + add()); } catch (ArithmeticException e) { e.printStackTrace(); }
通过throws抛出异常时,针对可能出现的多种异常情况,解决方案:
1. throws后面接多个异常类型,中间用逗号分隔2. throws后面接Exception,直接Exception的时候,编译器会提示你test方法可能会产生异常,因为Exception是父类包含了出现检查异常的情况。
而我们只写几个具体的异常类型(非检查型异常时)时是不会有提示的,如果想要提醒别人注意添加异常处理,可以添加文档注释。
/** * 两数字相加的方法 * @return * @throws ArithmeticException * @throws InputMismatchException */
throw抛出异常对象
throw用来抛出一一个异常。
例如: throw new IOException();
throw抛出的只能够是可抛出类Throwable或者其子类的实例对象。
例如: throw new String("出错啦");
是错误的
public void method(){ try { //代码段1 throw new 异常类型(); } catch(异常类型 ex){ //对异常进行处理的代码段2 } }
方案一: 自己抛出的异常自己进行异常处理。
方案二: 在抛出异常处通过throws关键字标明异常类型。
public void method() throws 异常类型{ //代码段1 throw new 异常类型(); }
规避可能出现的风险; 2. 完成一些程序的逻辑
场景: 要求:如果入住人员年龄在18岁以下或者80岁以上,必须由亲属陪同入住,不能单独入住。
public static void testAge() { try { System.out.println("请输入年龄:"); Scanner input = new Scanner(System.in); int age = input.nextInt(); if (age < 18 || age > 80) { throw new Exception("18岁以下,80岁以上的住客必须由亲友陪同"); } else { System.out.println("欢迎入住本酒店"); } } catch (Exception e) { e.printStackTrace(); } }
public static void main(String[] args) { testAge(); }
运行结果:
throw抛出异常对象的处理方案:
1. 通过try..catch包含throw语句-自己抛自己处理2. 通过throws在方法声明出抛出异常类型-谁调用谁处理-调用者可以自己处理,也可以继续上抛,此时可以抛出与throw对象相同的类型或者其父类
public static void testAge() throws Exception { System.out.println("请输入年龄:"); Scanner input = new Scanner(System.in); int age = input.nextInt(); if (age < 18 || age > 80) { throw new Exception("18岁以下,80岁以上的住客必须由亲友陪同"); } else { System.out.println("欢迎入住本酒店"); } }
try { testAge(); } catch (Exception e) { e.printStackTrace(); }
关于throw抛出异常类型问题的逼叨叨
public static void testAge() throws Throwable { System.out.println("请输入年龄:"); Scanner input = new Scanner(System.in); int age = input.nextInt(); if (age < 18 || age > 80) { throw new Exception("18岁以下,80岁以上的住客必须由亲友陪同"); } else { System.out.println("欢迎入住本酒店"); } }
try { testAge(); } catch (Throwable e) { e.printStackTrace(); }
因为Throwable是Exception异常的父类。throws后面不能是Exception的子类。
此时可以抛出与throw对象相同的类型或者其父类(不能是子类)
public static void testAge(){ System.out.println("请输入年龄:"); Scanner input = new Scanner(System.in); int age = input.nextInt(); if (age < 18 || age > 80) { throw new ArithmeticException();// throw new Exception("18岁以下,80岁以上的住客必须由亲友陪同"); } else { System.out.println("欢迎入住本酒店"); } }
非检查异常,不做强制处理。
testAge(); //无提示
throw抛出异常时是不推荐抛出非检查异常的。
自定义异常
使用Java内置的异常类可以描述在编程时出现的大部分异常情况。
也可以通过自定义异常描述特定业务产生的异常类型。
所谓自定义异常,就是定义一个类,去继承Throwable类或者它的子类。
新建一个类: HotelAgeException 继承自Exception类
package cn.mtianyan.exception;public class HotelAgeException extends Exception{ public HotelAgeException(){ super("18岁以下,80岁以上的住客必须由亲友陪同"); } }
public static void testAge() throws HotelAgeException { System.out.println("请输入年龄:"); Scanner input = new Scanner(System.in); int age = input.nextInt(); if (age < 18 || age > 80) { throw new HotelAgeException(); } else { System.out.println("欢迎入住本酒店"); } }
因为是继承自Exception类的,会提醒throws。
try { testAge(); } catch (HotelAgeException e) { System.out.println(e.getMessage()); System.out.println("酒店前台工作人员不允许为其办理入住登记"); }
运行结果:
当然后面可以多添加几个catch来捕获。
异常链
有时候我们会捕获一个异常后在抛出另一个异常
后一个方法接收前一个方法抛出的异常对象。
package cn.mtianyan.exception;public class TryDemoFive { public static void main(String[] args) { try { testThree(); } catch (Exception e) { e.printStackTrace(); } } public static void testOne() throws HotelAgeException { throw new HotelAgeException(); } public static void testTwo() throws Exception { try { testOne(); } catch (HotelAgeException e) { throw new Exception("我是新产生的异常1"); } } public static void testThree() throws Exception { try { testTwo(); } catch (Exception e) { throw new Exception("我是新产生异常2"); } } }
最后只会获得最后一个方法调用时throw的异常,前面的异常都丢失了。如何解决这种异常的丢失情况,让异常链的情况为我们所知。
构造方法中添加上一层异常e,或者使用initCause(e)
public static void testOne() throws HotelAgeException { throw new HotelAgeException(); } public static void testTwo() throws Exception { try { testOne(); } catch (HotelAgeException e) { throw new Exception("我是新产生的异常1",e); } } public static void testThree() throws Exception { try { testTwo(); } catch (Exception e) { Exception e1 = new Exception("我是新产生异常2"); e1.initCause(e); throw e1; } }
作者:天涯明月笙
链接:https://www.jianshu.com/p/18a79f4e9c17