异常

 Java代码在运行时期发生的问题就是异常。

Java中,把异常信息封装成了一个类。当出现了问题时,就会创建异常类对象并抛出异常相关的信息(如异常出现的位置、原因等)。

1异常的继承体系

1.1使用Exception类来描述异常。

 

Exception的父类是Throwable。

Throwable是Java 语言中所有错误或异常的超类,即祖宗类。

 

1.2特殊子类:RuntimeException子类

RuntimeException及其它的子类只能在Java程序运行过程中出现。

 

1.3 Error

与异常Exception平级的有一个Error,它是Throwable的子类,它用来表示java程序中可能会产生的严重错误。解决办法只有一个,修改代码避免Error错误的产生。

 

异常继承体系总结:

Throwable: 它是所有错误与异常的超类(祖宗类)

|- Error 错误

|- Exception 编译期异常,进行编译JAVA程序时出现的问题

|- RuntimeException 运行期异常, JAVA程序运行过程中出现的问题

 

2异常与错误的区别

例1

 

 

一般第一个位置是主要原因。

2

 

内存溢出错误:

 

异常:感冒发烧,是可以治的。可以对异常进行具体的处理。若不处理异常,程序将会结束运行。

错误:癌症,艾滋病,是不能治的,一旦发生错误,无法针对处理,只能修改代码。

3异常产生的过程

1)Jvm检测到异常,然后创建一个异常对象,

2)将该对象向上抛出,抛给调用者,

3)调用者接收到异常后,如果不能处理,就继续向上抛,抛给调用者,(到达main方法,最后到达jvm),

4)如果最后抛给了jvm,就将该异常对象及信息以红字的形式打印到控制台,并终止程序 

注意:一旦产生异常,下面的代码将不再执行。 

图说明:

 

4抛出异常throw

在编写程序时,我们必须要考虑程序出现问题的情况。比如,在定义方法时,方法需要接受参数。那么,当调用方法使用接受到的参数时,首先需要先对参数数据进行合法的判断,数据若不合法,就应该告诉调用者,传递合法的数据进来。这时需要使用抛出异常的方式来告诉调用者。 

java中,提供了一个throw关键字,它用来抛出一个指定的异常对象。

使用格式:

throw new 异常类名(参数);

5声明异常throws

声明:将问题标识出来,报告给调用者。

声明异常格式:

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

例:(把异常发给jvm,一般不这样,因为会终止程序)

public class Test2 {
	public static void main(String[] args) throws Exception{
		int[] arr={};
		int i=get(arr);
		System.out.println(i);
	}
	
	//抛异常
	public static int get(int[] arr) throws Exception{
		if(arr.length==0){
			throw new Exception("传入空数组");
		}
		if(arr==null){
			throw new Exception("传入数组为null");
		}
		
		int i=arr[arr.length-1];
		return i;
	}
}

6捕获异常(调用者处理)

捕获:Java中对异常有针对性的语句进行捕获,可以对出现的异常进行指定方式的处理

捕获异常格式:

try {

//需要被检测的语句。

}

catch(异常类 变量) { //参数。

//异常的处理语句。

}

finally {

//一定会被执行的语句。

}

try:该代码块中编写可能产生异常的代码。

catch:用来进行某种异常的捕获,实现对捕获到的异常进行处理。

finally:有一些特定的代码无论异常是否发生,都需要执行。另外,因为异常会引发程序跳转,导致有些语句执行不到。而finally就是解决这个问题的,在finally代码块中存放的代码都是一定会被执行的。(可以不写) 

例:

public class Test3 {
	public static void main(String[] args){
		int[] arr={};
		
		try{
			int i=get(arr);
			System.out.println(i);
		}catch(Exception ex){
			System.out.println(ex);
		}finally{
			System.out.println("一定会执行的代码");
		}
		
		System.out.println("异常外的代码");
	}
	
	public static int get(int[] arr) throws Exception{
		if(arr.length==0){
			throw new Exception("不能传入空数组");
		}
		if(arr==null){
			throw new Exception("传入数组为null");
		}
		
		int i=arr[arr.length-1];
		return i;
	}
}

 

ex可以理解为形参,就是把new Exception传进来

注意:

1有多个异常,就要写多个catch

2)捕获时要有顺序,父类写在后面

例:

选中类,ctrl+T

 

7异常处理的组合方式

try catch finally组合:检测异常,并传递给catch处理,并在finally中进行资源释放。 

try catch组合 : 对代码进行异常检测,并对检测的异常传递给catch处理。对异常进行捕获处理。(组合块以外的代码还会执行) 

一个try 多个catch组合 : 对代码进行异常检测,并对检测的异常传递给catch处理。对每种异常信息进行不同的捕获处理。

这种异常处理方式,要求多个catch中的异常不能相同,并且若catch中的多个异常之间有子父类异常的关系,那么子类异常要求在上面的catch处理,父类异常在下面的catch处理。 

try  finally 组合: 对代码进行异常检测,检测到异常后因为没有catch,所以一样会被默认jvm抛出。异常是没有捕获处理的。但是功能所开启资源需要进行关闭,所有finally。只为关闭资源。(组合块以外的代码不会执行)

 

8运行时期异常

编译时期异常:Exception及其子类(但不包括RuntimeException和其子类)

运行时期异常:RuntimeException和其子类 

RuntimeException和他的所有子类异常,都属于运行时期异常。

NullPointerException,ArrayIndexOutOfBoundsException等都属于运行时期异常。

特点:

方法中抛出运行时期异常,方法定义中无需throws声明,调用者也无需处理此异常,

运行时期异常一旦发生,需要程序人员修改源代码。 

例:没有报红(错误)

编译失败,因为parse方法里面定义并抛出了异常。查看一下:

 

这时有两种处理方式:

1thorws,2try 

可以直接点出来:

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class DateTest {
	public static void main(String[] args) {
		SimpleDateFormat sdf=new SimpleDateFormat("");
		try {
			Date d=sdf.parse("2018年11月11日");
		} catch (ParseException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		System.out.println("异常外的代码");
	}
}

就不会编译失败了

e.printStackTrace();这个方法就是打印jvm的信息,但不会终止程序 

 

实际应用:当遇到jdk里面一些类的异常时,如果编译失败,那么会处理就可以(两种方式)。

 

9异常在方法重写中细节

1)如果父类方法抛出异常,子类继承父类重写该方法时,要么不抛出异常,如果抛,必须抛父类方法抛出的异常或其子类。

  

2)如果父类方法没有抛出异常,那么子类继承父类重写该方法时,不允许抛出任何异常。

 

这时,只能用try

 

10异常中常用方法

getMessage方法:返回该异常的简短描述

toString方法:返回该异常的名称与详细信息字符串

printStackTrace:在控制台输出该异常的名称与详细信息字符串、异常出现的代码位置,红字打印异常信息(最常用)

例:

public class Test1 {
	public static void main(String[] args) {
		int[] arr={};
		int i=0;
		try {
			i = get(arr);
		} catch (Exception e) {	
			System.out.println(e.getMessage());
			System.out.println(e.toString());
			e.printStackTrace();			
		}
		System.out.println(i);
	}
	
	public static int get(int[] arr) throws Exception{
		if(arr.length==0){
			throw new Exception("数组为空");
		}
		return arr.length-1;
	}
}

11自定义异常

可以模拟Java的这种机制,我们自己定义异常的信息,异常的名字,让异常更符合自己程序的阅读。准确对自己所需要的异常进行类的描述。

例:

public class FuShuException extends RuntimeException{
	public FuShuException(){
		
	}
	public FuShuException(String str){
		super(str); //调用父类构造,并传入参数
	}
}
public class ScoreTest {
	public static void main(String[] args) {
		int[] arr={1,2,-3,4,5};
		double score=avg(arr);
		System.out.println(score);
	}
	//求学生平均成绩
	public static double avg(int[] arr){
		int sum=0;
		for(int i:arr){
			if(i<0){
				throw new FuShuException("传入的数组中有负数"+i);
			}
			sum+=i;
		}
		return sum/arr.length;
	}
}

原文地址:https://www.cnblogs.com/hzhjxx/p/10205417.html