异常处理技巧

java的异常处理涉及到程序流程的跳转,所以,虚拟机需要保存程序的执行流程,以便异常发生时能正确的跳转,这也就导致了使用异常时会引起额外的开销,所以,要谨慎地使用异常。

使用异常有如下几个原则:
1、异常不能代替简单测试,使用异常会引起额外开销。
2、不要为每个可能会出现异常的语句都设置try和catch。
3、避免在方法中抛出(throw)或者捕获(catch)运行时异常RuntimeException和Error。
4、避免总是catch Exception或Throwable,而要catch具体的异常类。这样可以使程序更加清晰。
5、不要压制、隐瞒异常。将不能处理的异常往外抛,而不是捕获之后随便处理。
6、不要在循环中使用try...catch,尽量将try...catch放在循环外或者避免使用。

import  java.util.Date;
import  java.util.EmptyStackException;
import  java.util.Stack;
public   class  ExceptionTips  {
    
    public static void main(String[] args) {
        
        //(1)异常不能代替简单测试
        Stack<Object> stack = new Stack();
        try{
            stack.pop();
        } catch (EmptyStackException e) {
            //....
        }
        //应该用下面的方式,以避免使用异常
        if (!stack.isEmpty()) {
            stack.pop();
        }
        
        //(2)不要为每个可能会出现的一场的语句都设置try和catch
        try{
            stack.pop();
        } catch (EmptyStackException e) {
            //....
        }
        String data = "123";
        try {
            Double.parseDouble(data);
        } catch(NumberFormatException e){
            //....
        }
        //应该使用下面的方式,将两个语句放在一个try块中
        try{
            stack.pop();
            Double.parseDouble(data);
        } catch(EmptyStackException e) {
            //....
        } catch(NumberFormatException e) {
            //....
        }
        
        //(3)避免在方法中抛出或者捕获运行时异常RuntimeException和Error,
        //比如内存错误等
        //避免出现下面的情况
        String[] array;
        try {
            array = new String[1000];
            //array = new String[1000000];此时会出现OutOfMemoryError异常
        } catch (OutOfMemoryError e) {
            throw e;
        }
        //直接用下面代码
        array = new String[1000];
        
        //(4)避免总是catch Exception或Throwable,而要捕获具体的异常
        //这样可以根据不同的异常做不同的处理,使程序更加清晰
        try {
            stack.pop();
            Double.parseDouble(data);
        } catch (Exception e) {
            //应该避免catch Exception !!!
        }
        
        //(5)不要压制、隐瞒异常。将不能处理的异常往外抛,而不是捕获之后随便处理
        try{
            Double.parseDouble(data);
        } catch (NumberFormatException e) {
            //.....
            throw e;    //抛出不能处理的异常,而不是隐瞒
        }
        
        //(6)不要在循环中使用try catch,尽量将try catch放在循环外或者避免使用try catch
        //下面的例子在循环中使用try和catch将耗费更多的时间,尽管没有异常发生
        int i = 0;
        int ntry = 1000000;
        Stack s = new Stack();
        long s1;
        long s2;
        System.out.println("Testing for empty stack");
        s1 = new Date().getTime();
        for (i = 0; i <= ntry; i++) {
            if (!s.empty()) {
                s.pop();
            }
        }
        s2 = new Date().getTime();
        System.out.println((s2 - s1) + "milliseconds");
        
        System.out.println("Catching EmptyStackException");
        s1 = new Date().getTime();
        for(i = 0; i<=ntry; i++) {
            try {
                s.pop();
            } catch(EmptyStackException e) {
            }
        }
        s2 = new Date().getTime();
        System.out.println((s2 - s1) + "milliseconds");
    }
}