天天看点

异常详解及自定义异常和注意事项

异常的概述

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

异常的分类

通过API查看Throwable      
1.Error
		 服务器宕机,数据库崩溃等
     2.Exception      

异常的继承体系

* Throwable
		* Error	
		* Exception
			* RuntimeException运行时异常,我们最容易犯的错误      

JVM默认是如何处理异常的

main函数收到这个问题时,有两种处理方式:
		a:自己将该问题处理,然后继续运行
		b:自己没有针对的处理方式,只有交给调用main的jvm来处理
			* jvm有一个默认的异常处理机制,就将该异常进行处理.
			* 并将该异常的名称,异常的信息.异常出现的位置打印在了控制台上,同时将程序停止运行      

测试异常处理

public static void main(String[] args) {
		// TODO Auto-generated method stub
		int x = 10 / 0;
		System.out.println(x);
		//Exception in thread "main" java.lang.ArithmeticException: / by zero
	}
           
public class ArithmeticExceptionextends RuntimeException
	 当出现异常的运算条件时,抛出此异常。例如,一个整数“除以零”时,抛出此类的一个实例。       

异常处理的两种方式

try…catch…finally

try catch
	try catch finally
	try finally       

throws

try...catch处理异常的基本格式
			try…catch…finally      
try:用来检测异常的
       
catch:用来捕获异常的
finally:释放资源
当通过trycatch将问题处理了,程序会继续执行      
public static void main(String[] args) {
		try{
			int x = 10/0;
			System.out.println(x);
		}catch(ArithmeticException e) {	
			//ArithmeticException a = new ArithmeticException();
			System.out.println("出错了,除数为零了");
			System.out.println(e);//java.lang.ArithmeticException: / by zero
		}
	}
           

try...catch的方式处理多个异常

JDK7以后处理多个异常的方式及注意事项	  
		安卓,客户端开发,如何处理异常?try{}catch(Exception e){}
	 	ee,服务端开发,一般都是底层开发,从底层向上抛
	 try后面如果跟多个catch,那么小的异常放前面,大的异常放后面,根据多态的原理,如果大的放前面,就会将所有的子类对象接收
  后面的catch就没有意义了      
public static void main(String[] args) {
		int a = 10;
		int b = 0;
		int[] arr = {11,22,33,44,55};
		
		try {
			System.out.println(a / b);
			System.out.println(arr[10]);
			arr = null;
			System.out.println(arr[0]);
		} catch (ArithmeticException e) {
			System.out.println("除数不能为零");
		} catch (ArrayIndexOutOfBoundsException e) {
			System.out.println("索引越界了");
		} catch (Exception e) {	//Exception e = new NullPointerException();
			System.out.println("出错了");
		}
		System.out.println("over");
		
		
		int a1 = 10;
		int b1= 0;
		int[] arr1 = {11,22,33,44,55};
		//JDK7如何处理多个异常
		try {
			System.out.println(a1 / b1);
			System.out.println(arr1[10]);
		} catch (ArithmeticException | ArrayIndexOutOfBoundsException e) {
			System.out.println("出错了");
		} 
	}
           

编译期异常和运行期异常的区别

* Java中的异常被分为两大类:编译时异常和运行时异常。
	* 所有的RuntimeException类及其子类的实例被称为运行时异常,其他的异常就是编译时异常
	* 编译时异常
		* Java程序必须显示处理,否则程序就会发生错误,无法通过编译
	* 运行时异常
		* 无需显示处理,也可以和编译时异常一样处理      

Throwable的几个常见方法

getMessage()
				* 获取异常信息,返回字符串。
		:toString()
				* 获取异常类名和异常信息,返回字符串。
		printStackTrace()
				* 获取异常类名和异常信息,以及异常出现在程序中的位置。返回值void。      
public static void main(String[] args) {
		try {
			System.out.println(1/0);
		} catch (Exception e) {	
			//Exception e = new ArithmeticException("/ by zero");
			System.out.println(e.getMessage());	//获取异常信息/ by zero
			System.out.println(e); 	//调用toString方法,打印异常类名和异常信息
			//java.lang.ArithmeticException: / by zero
			e.printStackTrace();//jvm默认就用这种方式处理异常
			//java.lang.ArithmeticException: / by zero
			//at a.main(a.java:112)
		}
	}
           

throws的方式处理异常

*定义功能方法时,需要把出现的问题暴露出来让调用者去处理。
那么就通过throws在方法上标识。
案例演示
	* 举例分别演示编译时异常和运行时异常的抛出
		* 编译时异常的抛出必须对其进行处理
		* 运行时异常的抛出可以处理也可以不处理
 @throws Exception       
public class a {
	public static void main(String[] args) throws Exception {
		Person p = new Person();
		p.setAge(-17);
		System.out.println(p.getAge());
	}	
}
class Person {
	private String name;
	private int age;
	public Person() {
		super();
		
	}
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	/*public void setAge(int age) throws Exception {
		if(age >0 && age <= 150) {
			this.age = age;
		}else {
			//Exception e = new Exception("年龄非法");
			//throw e;
			throw new Exception("年龄非法");
		}
	}*/
	public void setAge(int age)  {
		if(age >0 && age <= 150) {
			this.age = age;
		}else {
			throw new RuntimeException("年龄非法");
		}
	}
}
           

throw的概述以及和throws的区别

throw的概述

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

throws和throw的区别

:throws
		* 用在方法声明后面,跟的是异常类名
		* 可以跟多个异常类名,用逗号隔开
		* 表示抛出异常,由该方法的调用者来处理
	throw
		* 用在方法体内,跟的是异常对象名
		* 只能抛出一个异常对象名
		* 表示抛出异常,由方法体内的语句处理      

finally的特点及作用

特点

被finally控制的语句体一定会执行
	特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))      

作用

用于释放资源,在IO流操作和数据库操作中会见到		      
return语句在执行之前看一看有没有finally,如果有就将finally执行后在彻底返回      
public class b {
	public static void main(String[] args) {
		try {
			System.out.println(10/0);
		} catch (Exception e) {
			System.out.println("除数为零了");
			//System.exit(0);//退出jvm虚拟机
			return;
		} finally {
			System.out.println("看看我执行了吗");
		}
	}
}
           

finally关键字的面试题

面试题1 final,finally和finalize的区别      
final可以修饰类,不能被继承
		final可以修饰方法,不能被重写
		final可以修饰变量,只能赋值一次
		 
		finally是try语句中的一个语句体,不能单独使用,用来释放资源

		finalize是一个方法,当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
			protected void finalize() throws Throwable
			当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
			子类重写 finalize 方法,以配置系统资源或执行其他清除。 
      

面试题2    如果catch里面有return语句,请问finally的代码还会执行吗?如果会,请问是在return前还是return后。

先建立好返回路径,然后执行finally的代码,最后彻底返回      
public class c {
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Demo d = new Demo();
		System.out.println(d.method());//30
	}

}

class Demo {
	//@SuppressWarnings("finally")
	public int method() {
		int x = 10;
		try {
			x = 20;
			System.out.println(1/0);
			//return x;
			return x;
		} catch (Exception e) {
			x = 30;
			return x;
		} finally {
			x = 40;//catch已经建立好返回路径,在此改变x的值不会影响catch的返回值
			//return x;//40
			//千万不要在finally里面写返回语句,因为finally的作用是为了释放资源,是肯定会执行的
			//如果在这里面写返回语句,那么try和catch的结果都会被改变,所以这么写就是犯罪
		}
	}
}
           

自定义异常概述和基本使用

为什么需要自定义异常

易于排错,望名生义(取决于名字)      

自定义异常概述

继承自Exception
	继承自RuntimeException
	重写异常的空参有参构造,参数向上传递super      

自定义异常的基本使用

public class a {  
    public static void main(String[] args) throws Exception {  
        Person p = new Person();  
        p.setAge(-17);  
        System.out.println(p.getAge());  
    }     
}  
/*
 * 自定义异常
 * 也可以继承 RuntimeException运行时异常,编译时不用处理
 * */
class AgeOutOfBoundsException extends Exception {
	public AgeOutOfBoundsException() {
		super();	
	}

	public AgeOutOfBoundsException(String message) {
		super(message);	
	}
}

class Person {  
    private String name;  
    private int age;  
    public Person() {  
        super();  
          
    }  
    public Person(String name, int age) {  
        super();  
        this.name = name;  
        this.age = age;  
    }  
    public String getName() {  
        return name;  
    }  
    public void setName(String name) {  
        this.name = name;  
    }  
    public int getAge() {  
        return age;  
    } 
    //异常抛出处理,查看Exception
    public void setAge(int age) throws AgeOutOfBoundsException  {  
        if(age >0 && age <= 150) {  
            this.age = age;  
        }else {  
          // throw new AgeOutOfBoundsException();  
        	throw new AgeOutOfBoundsException("年龄错误");
        }  
    }  
}  
           

异常的注意事项及如何使用异常处理

异常注意事项

a:子类重写父类方法时,子类的方法必须抛出相同的异常或父类异常的子类。
	b:如果父类抛出了多个异常,子类重写父类时,只能抛出相同的异常或者是他的子集,子类不能抛出父类没有的异常
	c:如果被重写的方法没有异常抛出,那么子类的方法绝对不可以抛出异常,
	     如果子类方法内有异常发生,那么子类只能try,不能throws      

如何使用异常处理

原则:如果该功能内部可以将问题处理,用try,如果处理不了,交由调用者处理,这是用throws
	区别:
		后续程序需要继续运行就try
		后续程序不需要继续运行就throws      

如果JDK没有提供对应的异常,需要自定义异常。