除了自定义类,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