1.异常概述

  异常就是java程序在运行过程中出现的错误。


2.异常的由来

  问题也是现实生活中一个具体事物,也可以通过java类的形式进行描述,并封装成对象。其实java对不正常的情况进行描述后的对象体现。


3.异常例子

  坐标越界异常,空指针异常。


4.异常图解

java之异常_java




异常分为三类:

Error:用Error进行描述,这个问题发生后,一般不编写针对代码进行处理,而是要对程序进行修正,通常都是由虚拟机内部抛出的问题。


RuntimeException:运行时异常,这种异常不用处理,编译会通过,不过这样的程序会有安全隐患,遇到这种异常是需要修改代码的。


非RuntimeException:除了RuntimeException及其子类,Exception中的所有子类都是,这种异常必须处理,不然编译通不过。


5.java的默认处理异常机制

package cn;

public class ExceptionDemo {
	public static void main(String[] args) {
		int a = 10;
		int b = 0;
		
		System.out.println(a /b);
		
		System.out.println("over");
		
		
	}
}

Exception in thread "main" java.lang.ArithmeticException: / by zero

at cn.ExceptionDemo.main(ExceptionDemo.java:8)


java的默认处理方案:

  把异常的名称,错误原因及异常出现的位置等信息输出在了控制台。

  程序停止了执行。


上面异常处理的过程是:

  java发现运算时违反了数学运算规则,java将这种常见的问题进行描述,并封装成了对象叫做ArithmeticException。

  当除以0运算发生后,java虚拟机将该问题打包成一个异常对象。

  并把这个对象抛给调用者main方法,new ArithmeticException("/ by zero")


  main方法受到这个问题,有两种处理方式。

  1.自己将该问题处理,然后继续运行。

  2.自己没有针对的处理方式,只有交给调用main方法的java虚拟机来处理。


6.异常处理方案


6.1 处理格式一:try..catch...finally

try{

  //检测异常

}catch(){

  //用来捕获异常

}finally{

  //用来结束资源的

}


这种异常的好玩的句子就是:世界上最真情的相依,是你在try我在catch,无论你发神马脾气,我都默默接收,静静处理。

package cn;

public class ExceptionDemo {
	public static void main(String[] args) {
		
		int a = 10;
		int b = 0;
		try{
			System.out.println(a /b);
		}catch(Exception e){
			e.printStackTrace();
		}
		
		System.out.println("over");
		
		
	}
}


6.2throws处理异常

  有些时候,我们是可以进行处理的,但是又有些时候,我们根本没有权限去处理某个异常。或者说,我处理不了,我就不处理了。

  为了解决出错问题,java针对这种情况,就提供了另一种解决方案:抛出。

格式:

  throws 异常类名

  这个格式必须跟在方法的括号后面。

package cn;

import java.text.ParseException;
import java.text.SimpleDateFormat;

public class ExceptionDemo {
	public static void main(String[] args) {
		System.out.println("今天天气很好");
		try {
			method();
		} catch (ParseException e) {
			e.printStackTrace();
		}
		
		System.out.println("为什么要有雾霾");
		
		
	}
	public static void method() throws ParseException{
		String s = "2016-11-11";
		System.out.println(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").parse(s));
	}
}

今天天气很好

为什么要有雾霾

java.text.ParseException: Unparseable date: "2016-11-11"

at java.text.DateFormat.parse(DateFormat.java:366)

at cn.ExceptionDemo.method(ExceptionDemo.java:21)

at cn.ExceptionDemo.main(ExceptionDemo.java:10)


7.编译时异常和运行时异常的区别

  java中的异常被分为两大类:编译时异常和运行时异常。所有的RuntimeException类及其子类的实例被称为运行时异常,其他的异常就是编译时异常。

  编译时异常:java程序必须显示处理,否则程序就会发生错误,无法通过编译。

  运行时异常:无需显示处理,也可以和编译时异常一样处理。


8.Throwable中的方法

getMessage(): 获取异常信息,返回字符串

toString(): 获取异常类名和异常信息,返回字符串

printStackTrace():获取异常类型和异常信息,以及异常出现在程序中的位置,返回值类型void

printStackTrace(PrintStream s):通常用该方法将异常内容保存在日志文件中,以便查阅。


9.throws关键字

  定义功能方法的时候,需要把出现的问题暴露出来让调用者去处理,那么就通过throws在方法上标识。


10.throw

  在功能方法内部出现某种情况,程序不能继续运行,需要进行跳转时,就用throw把异常对象抛出。

package cn;
/**
 * throw:如果出现了异常情况,我们可以将该异常抛出,这个时候抛出的应该是异常的对象 
 *
 */
public class ExceptionDemo2 {
	public static void main(String[] args) {
		method(0);
	}
	public static void method(int b){
		int a = 10;
		if(b == 0){
			throw new ArithmeticException("除数不能为0");
		}else{
			System.out.println(a / b);
		}
		
	}

}

Exception in thread "main" java.lang.ArithmeticException: 除数不能为0

at cn.ExceptionDemo2.method(ExceptionDemo2.java:13)

at cn.ExceptionDemo2.main(ExceptionDemo2.java:8)

package cn;
/**
 * throw:如果出现了异常情况,我们可以将该异常抛出,这个时候抛出的应该是异常的对象 
 *
 */
public class ExceptionDemo2 {
	public static void main(String[] args) {
		try {
			method(0);
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
	public static void method(int b) throws Exception{
		int a = 10;
		if(b == 0){
			throw new Exception("除数不能为0");
		}else{
			System.out.println(a / b);
		}
		
	}

}

java.lang.Exception: 除数不能为0

at cn.ExceptionDemo2.method(ExceptionDemo2.java:17)

at cn.ExceptionDemo2.main(ExceptionDemo2.java:9)


两段程序片段,结果差不多,为什么形式大不同?

  第一个程序片段中,抛出了是RuntimeException的对象,是可以不处理的啊。

  但是在第二个程序片段中,抛出了是Exception,这时java就认为了你必须处理,否则不让你编译通过哦。

  而抛出的是RuntimeException,在方法上是可以不写throws的,因为是运行时异常,可以不处理。

  但是一旦抛出的是非RuntimeException异常,那么恭喜你,要么直接处理,要么在方法的括号后写上throws 异常类名吧,因为此时java认为你必须处理。


11.throws和throw的区别?

throws

  用在方法声明后面,跟的是异常类名。

  可以跟多个异常类名,用逗号隔开。

  表示抛出异常,由该方法的调用者处理。

  throws表示出现异常的一种可能性,并不一定会发生这些异常。


throw

  用在方法内,跟的是异常对象名。

  只能抛出一个异常对象名。

  表示抛出异常,由方法体内的语句进行处理。

  throw则是抛出了异常,执行throw则一定抛出了某种异常。


12.我们该如何处理异常?

原则:如果该功能内部可以将问题处理,用try,如果处理不了,交由调用者处理,就用throws.

区别:

  后续程序需要继续运行就try。

  后续程序不需要继续运行就throws。


13.finally

finally的特点:

  被finally控制的语句体一定会执行。

  但是如果在执行到finally之前JVM就退出了,那么finally就不起作用了,即此时被finally控制的语句体不会执行。

finally作用

  用于释放资源


14.final,finally和finaalize的区别?

final是最终的意思,可以修饰类,成员变量和成员方法。

  final修饰类,类不能被继承。

  final修饰变量,变量是常量。

  final修饰方法,方法不能被重写。

finally:是异常处理的一部分,用于释放资源。

  一般来说,代码肯定会执行,但是,如果在执行到finally之前,JVM退出了,怎么还可能执行呢?

finalize:是Object类的一个方法,用于垃圾回收。


15.自定义异常

  java不可能对所有的情况都考虑到,所以,在实际的开发中,我们可能需要自己定义异常。而我们随意的一个类,是不能作为异常类来看的,要想你的类是一个异常类,就必须继承自Exception或RuntimeException。

package cn;

@SuppressWarnings("serial")
public class MyException extends Exception{
	public MyException(){}
	public MyException(String message){
		super(message);
	}
}


16.异常的注意事项

16.1  子类重写父类方法时,子类的方法必须抛出相同的异常或者父类异常的子类。

class Fu{
	
	public void method() throws Exception{
		
	}
}
class Zi extends Fu{
	public void method() throws Exception{
		
	}
}
class Fu{
	
	public void method() throws Exception{
		
	}
}
class Zi extends Fu{
	public void method() throws ArithmeticException{
		
	}
}

为什么呢?因为子类重写父类的方法,是因为子类需要扩充父类的功能,既然要扩展父类的功能,那么子类抛出的异常要么和父类相同,要么就比父类具体。(父亲坏,儿子不能比父亲更坏吧,不然,会被人家骂,上梁不正下梁歪)


16.2  如果父类抛出了多个异常,子类重写父类时,只能抛出相同的异常或者他的子集,子类不能抛出父类没有的异常。


16.3  如果被重写的方法没有异常抛出,那么子类的方法绝对不可以抛出异常,如果子类方法内有异常发生,那么子类只能try,却不能throws。

class Fu{
	
	public void method() throws Exception{
		
	}
	
	public void method2(){
		
	}
}
class Zi extends Fu{
	public void method() throws ArithmeticException{
		
	}
	public void methods(){
		String str = "2016-11-11";
		try {
			Date d = new SimpleDateFormat("yyyy-MM-dd").parse(str);
			System.out.println(d);
		} catch (ParseException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}