• 2020.10.26收获+动手动脑五


    ******************************************蓝色粗体为动手动脑内容********************************************

      异常处理   

    1、Java异常处理基础

    异常:发生于程序执行期间,表明出现了一个非法的运行状况。许多哦JDK中的方法在检测到非法情况时,都会抛出一个异常对象。

    例如:数组越界和被零除

    AboutException.java

    import javax.swing.*;
    
    class AboutException {
       public static void main(String[] a) 
       {
          int i=1, j=0, k;
          k=i/j;
    try { k = i/j; // Causes division-by-zero exception //throw new Exception("Hello.Exception!"); } catch ( ArithmeticException e) //捕获算术异常 { System.out.println("被0除. "+ e.getMessage()); } catch (Exception e) { if (e instanceof ArithmeticException) System.out.println("被0除"); else { System.out.println(e.getMessage()); } } finally { JOptionPane.showConfirmDialog(null,"OK"); } } }

     注意:finally 无论有没有错误都执行,finally 是最后一道防线

    异常捕获语句

    注:Java中所有可捕获的异常都派生自Exception类

    使用Java异常处理机制:

    • 把可能会发生错误的代码放进try语句块中
    • 当程序检测到出现了一个错误时会抛出一个异常对象。异常处理代码会捕获并处理这个错误
    • catch语句块中的代码用于处理错误
    • 当异常发生时,程序控制流程由try语句块跳转到catch语句块
    • 不管是否有异常发证,finally语句块中的语句始终保证被执行
    • 如果没有提供合适的异常处理代码,JVM将会结束掉整个应用程序

     

    JDK中与异常相关的类

     Java中的异常分类:

    Throwable类有两个直接子类:

    • Exception:出现的问题是可以被捕获的
    • Error:系统错误,通常由JVM处理

      可捕获的异常又可以分为两类:

    • check异常:直接派生自Exception的异常类,必须被捕获或再次声明抛出
    • Runtime异常:派生自RUntimeException的异常类。使用throw语句可以随时抛出这种异常对象:throw new ArithmeticException(...)

    JDK1.4以上提供了assert语句,允许程序在运行期间判断某个条件是否满足,不满足时,抛出AssertionError

    注:默认情况下,assert功能是关闭的,可以在使用Java启动JVM时添加参数-ea打开它。

    异常的“多态”特性:

    • 可以有多个catch语句块,每个代码块捕获一种异常,在某个try块后有两个不同的catch块捕获两个相同类型的异常是语法错误
    • 使用catch语句,只能捕获Exception类及其子类的对象。因此,一个捕获Exception对象的catch语句块可以捕获所有“可捕获”的异常
    • 将catch(Exception e)放在别的catch块前面会使这些catch块都不指定,因此Java不会编译这个程序

    “finally"的功用

    • 资源泄露:当一个资源不再被某应用程序使用,但此程序并未向系统声明不再使用此资源时发生这种情况
    • finally语句块主要用于解决资源泄露问题,它位于catch语句块之后,JVM保证它们一定执行
    • 注:finally语句块中也可能发生异常,当这种情况发生,先前的异常被放弃

    多层的异常捕获-1(CatchWho.java)

    public class CatchWho 
    { 
        public static void main(String[] args) 
        { 
            try 
        { 
                    try 
            { 
                        throw new ArrayIndexOutOfBoundsException(); 
                    } 
                    catch(ArrayIndexOutOfBoundsException e) 
            { 
                           System.out.println(  "ArrayIndexOutOfBoundsException" +  "/内层try-catch"); 
                    }
     
                throw new ArithmeticException(); 
            } 
            catch(ArithmeticException e) 
        { 
                System.out.println("发生ArithmeticException"); 
            } 
            catch(ArrayIndexOutOfBoundsException e) 
        { 
               System.out.println(  "ArrayIndexOutOfBoundsException" + "/外层try-catch"); 
            } 
        } 
    }

    运行结果:

     CatchWho2.java

    public class CatchWho2 
    { 
        public static void main(String[] args) 
        { 
            try 
        {
                    try 
            { 
                        throw new ArrayIndexOutOfBoundsException(); 
                    } 
                    catch(ArithmeticException e) 
            { 
                        System.out.println( "ArrayIndexOutOfBoundsException" + "/内层try-catch"); 
                    }
                throw new ArithmeticException(); 
            } 
            catch(ArithmeticException e) {
         
                System.out.println("发生ArithmeticException"); 
            } 
            catch(ArrayIndexOutOfBoundsException e) 
        { 
                System.out.println( "ArrayIndexOutOfBoundsException" + "/外层try-catch"); 
            } 
        } 
    }

    运行结果:

     阅读EmbedFinally.java示例,在运行它,观察其输出并进行总结

    public class EmbededFinally {
    
        
        public static void main(String args[]) {
            
            int result;
            
            try {
                
                System.out.println("in Level 1");
    
               
                 try {
                    
                    System.out.println("in Level 2");
      // result=100/0;  //Level 2
                   
                     try {
                       
                         System.out.println("in Level 3");
                          
                         result=100/0;  //Level 3
                    
                    } 
                    
                    catch (Exception e) {
                        
                        System.out.println("Level 3:" + e.getClass().toString());
                    
                    }
                    
                    
                    finally {
                        
                        System.out.println("In Level 3 finally");
                    
                    }
                    
                   
                    // result=100/0;  //Level 2
    
                
                    }
                
                catch (Exception e) {
                   
                     System.out.println("Level 2:" + e.getClass().toString());
               
                 }
                 finally {
                    
                    System.out.println("In Level 2 finally");
               
                 }
                 
                // result = 100 / 0;  //level 1
            
            } 
            
            catch (Exception e) {
                
                System.out.println("Level 1:" + e.getClass().toString());
            
            }
            
            finally {
               
    .             System.out.println("In Level 1 finally");
            
            }
        
        }
    
    }

    运行结果:

     当有多个嵌套的try…catch…finally时,要特别注意finally的执行时机,当有多层嵌套的finally时,异常在不同的层次抛出 ,在不同的位置抛出,可能会导致不同的finally语句块执行顺序,当有多层嵌套的finally时,异常在不同的层次抛出 ,在不同的位置抛出,可能会导致不同的finally语句块执行顺序。

    finally一定会执行吗?请通过SystemExitAndFinally.java示例程序回答上述问题

    public class SystemExitAndFinally {
    
        public static void main(String[] args)
        {
            try{
    
                System.out.println("in main");
                throw new Exception("Exception is thrown in main");
                //System.exit(0);
    
            }
            catch(Exception e)
            {
                System.out.println(e.getMessage());
                System.exit(0);
            }
            finally
            {
                System.out.println("in finally");
            }
        }
    
    }

    运行结果:

    分析:在JVM正常运行的情况下,finally块一定会执行。但如果JVM都退出了finally块就无法执行了

    如何跟踪异常的传播路径?

    当程序中出现异常时,JVM会依据方法调用顺序依次查找有关的错误处理程序

    可使用printStackTrace和getMessage方法了解异常发生的情况:

    printStackTrace:打印方法调用堆栈

    每个Throwable类的对象都有一个getMessage方法,它返回一个字串,这个字串是在Exception构造函数中传入的,通常让这一字串包含特定异常的相关信息。

    PrintExpressionStack.java

    // UsingExceptions.java
    // Demonstrating the getMessage and printStackTrace
    // methods inherited into all exception classes.
    public class PrintExceptionStack 
    {
       public static void main( String args[] )
       {
          try 
          {
             method1();
          }
          catch ( Exception e ) 
          {
             System.err.println( e.getMessage() + "\n" );
             e.printStackTrace();
          }
       }
    
       public static void method1() throws Exception
       {
          method2();
       }
    
       public static void method2() throws Exception
       {
          method3();
       }
    
       public static void method3() throws Exception
       {
          throw new Exception( "Exception thrown in method3" );
       }
    }
  • 相关阅读:
    父子组件的数据传递
    前端处理:elementUI 表格索引代表第几条数据
    qt的moc,uic,rcc命令的使用
    支持无限精度无限大数的类BigNumber实现
    二进制的妙用
    C#实现任意大数的计算和简单逻辑命题的证明——前言
    打磨程序员的专属利器——文本
    打磨程序员的专属利器——快捷键
    打磨程序员的专属利器——命令行&界面
    Linux命令全集
  • 原文地址:https://www.cnblogs.com/ltw222/p/13885188.html
Copyright © 2020-2023  润新知