Java异常总结

Java异常总结

异常定义:

异常是一个发生在程序运行中的,能够中断正常业务逻辑的事件;

当方法运行当中出现错误的时候,方法会创建一个对象并且将其提交给Java运行环境,这个对象就叫做异常对象。异常对象包含了关羽这次错误的信息,以及其类型和当前错误发生的时候程序的运行状态。当方法抛出异常后,运行环境会尝试找到某种机制去处理他,这个可能的“机制”是java运行时候调用的方法栈。

  Java虚拟机会在抛出异常的方法调用栈中找到能够处理这个异常的代码,这代码块被称为异常处理代码。查找从当前方法开始一直往上一级方法查找。如果存在异常处理代码能够处理这种类型的异常,那么久被认为是和这个异常是匹配的。

 如果没有合适的处理代码来处理某种异常,java虚拟机会终止当前运行的进程。

 

 

 Java语言异常处理遵循着Catch or Specify 的规则,这就意味着某段能够抛出异常的代码必须以以下两种方式来进行处理

1.通过用try语句来处理该异常,try必须提供一个异常处理的方法

2.一个方法必须特化其将要抛出的异常,异常必须提供一个throw语句来对该异常进行抛出。

 

不遵循catch or specify的代码将不会通过编译。并不是所有的异常适用于catch or specify的规则。

 

异常主要是分为三种:

1.checked exception,代表了程序可以处理并且可以恢复的异常如java.io.FileNotFoundException

2.第二种异常是error 错误,这个异常不适用于catch or specify准则,程序可能会处理这些异常,但是最好的方法是终止进程并且找出程序当中的bug。

3.第三种异常是runtime exception。运行中的异常,通常这种异常出现时,程序不会从该异常中恢复。该类异常通常意味着程序当中的错误。例如不正确的使用api或者逻辑上的错误。例如如果给构造函数传递一个空指针之后,构造函数会抛出 NullPointerException。 Error错误和runtime exception异常被认为是unchecked exception;

那么如何处理异常呢,通常我们有两种异常的处理方法:一种是try,catch和finally这样的通用的异常处理模块。另外还有在java7中被引进的try-with-resource语句。

try {

 

} catch (ExceptionType name) {

 

} catch (ExceptionType name) {

 

}

Try catch 语句的形式。 Try语句中代表执行的代码段,代码中的异常发生在这里。对于每一个catch语句来说,catch语句是其参数所表明的该ExceptionType 异常的处理代码,java 虚拟机在执行异常处理的过程当中会选择第一个匹配该异常参数的catch代码来执行。所有的catch中的异常类型都需要继承Throwable 类。异常处理代码通常包含了该异常的处理方式以及恢复手段,如果当前异常不能够被处理,其还可以通过异常链的方法来对当前异常进行抛出,使得更高级别的异常处理代码能够处理该异常。

 Finally模块一般来说会在try所标记的代码模块中被执行。一般在finally模块中做一些事后清理的工作。不过值得注意的一点是如果finally模块在执行之前当前线程被try或者catch模块中所处理的异常弄退出了或者是被杀掉了,finally模块则可能不执行。

finally {

    if (out != null) {

        System.out.println("Closing PrintWriter");

        out.close();

    } else {

        System.out.println("PrintWriter not open");

    }

}

Finally是防止资源泄露的一个有效的手段,当程序在使用计算机资源的时候或者做内存分配的时候通过在finally代码中放置资源处理代码可以很大可能的防止资源泄露。

 

Try with resource 语句

Try-with-resource 语句U是一个由try生命的一个或者多个的资源。所谓的资源就是一个对象使用完成后必须将其关闭。Try-with-resource保证了这些资源在try语句结束后将被关闭。只要有对象继承了java.lang.AutoCloseable 这个接口的对象,包括继承了java.io.Closeable 的对象,都可以被当作资源使用。代码如下:

 

static String readFirstLineFromFile(String path) throws IOException {

    try (BufferedReader br =

                   new BufferedReader(new FileReader(path))) {

        return br.readLine();

    }

}

 

在这个例子中所谓的资源对象是一个BufferReader,在java1.7中,这个对象继承了java.lang.AutoCloseable,由于这个资源对象是放置在try with resource语句中,则保证了其不管出现异常与否在执行完try所包含的代码快后,这个资源一定会被释放。和try-catch-finally比较起来。

public static void writeToFileZipFileContents(String zipFileName,

                                           String outputFileName)

                                           throws java.io.IOException {

 

    java.nio.charset.Charset charset =

         java.nio.charset.StandardCharsets.US_ASCII;

    java.nio.file.Path outputFilePath =

         java.nio.file.Paths.get(outputFileName);

 

    // Open zip file and create output file with

    // try-with-resources statement

 

    try (

        java.util.zip.ZipFile zf =

             new java.util.zip.ZipFile(zipFileName);

        java.io.BufferedWriter writer =

            java.nio.file.Files.newBufferedWriter(outputFilePath, charset)

    ) {

        // Enumerate each entry

        for (java.util.Enumeration entries =

                                zf.entries(); entries.hasMoreElements();) {

            // Get the entry name and write it to the output file

            String newLine = System.getProperty("line.separator");

            String zipEntryName =

                 ((java.util.zip.ZipEntry)entries.nextElement()).getName() +

                 newLine;

            writer.write(zipEntryName, 0, zipEntryName.length());

        }

    }

}

 

Java的方法抛出异常:如果当前方法会抛出某个异常,但是又不想在当前的方法进行处理,则最好在方法体中抛出异常。

public void writeList() throws IOException, IndexOutOfBoundsException {

其中IndexOutOfBoundsException 是一个unchecked的异常,是否抛出是可选的。

如何抛出异常:当你在处理异常之前,一定会有某段代码抛出了异常,抛出异常代码的部分可能会发生在你写的某段代码,或者某个你引用的jar包,你可以抛出自定义的异常。

 

异常链:有时候我们捕获到一个异常后,会抛出另外的一个新的异常。所以了解java中因为什么引起的这个异常是很有必要的,java的异常链机制能够实现这个目的。

Throwable接口中如下的方法会帮助实现异常链:

Throwable getCause()

Throwable initCause(Throwable)

Throwable(String, Throwable)

Throwable(Throwable)

 

 

使用异常链的代码如下所示:

try {

 

} catch (IOException e) {

    throw new SampleException("Other IOException", e);

}

 

如果一个高等级的异常希望dump当前的异常栈的信息,那么具体的代码示例如下:

catch (Exception cause) {

    StackTraceElement elements[] = cause.getStackTrace();

    for (int i = 0, n = elements.length; i < n; i++) {       

        System.err.println(elements[i].getFileName()

            + ":" + elements[i].getLineNumber()

            + ">> "

            + elements[i].getMethodName() + "()");

    }

}

 

其他还有很多异常,我就不一一列举了,我要说明的是,一个合格的程序员,需要对程序中常见的问题有相当的了解和相应的解决办法,否则仅仅停留在写程序而不会改程序的话,会极大影响到自己的开发的。关于异常的全部说明,在api里都可以查阅。 算术异常类:ArithmeticExecption 空指针异常类:NullPointerException 类型强制转换异常:ClassCastException 数组负下标异常:NegativeArrayException 数组下标越界异常:ArrayIndexOutOfBoundsException 违背安全原则异常:SecturityException 文件已结束异常:EOFException 文件未异常:FileNotFoundException 字符串转换为数字异常:NumberFormatException 操作数据库异常:SQLException 输入输出异常:IOException 方法未异常:NoSuchMethodException java.lang.AbstractMethodError 抽象方法错误。当应用试图调用抽象方法时抛出。 java.lang.AssertionError 断言错。用来指示一个断言失败的情况。 java.lang.ClassCircularityError 类循环依赖错误。在初始化一个类时,若检测到类之间循环依赖则抛出该异常java.lang.ClassFormatError 类格式错误。当Java虚拟机试图从一个文件中读取Java类,而检测到该文件的内容不符合类的有效格式时抛出。 java.lang.Error 错误。是所有错误的基类,用于标识严重的程序运行问题。这些问题通常描述一些不应被应用程序捕获的反常情况。 java.lang.ExceptionInInitializerError 初始化程序错误。当执行一个类的静态初始化程序的过程中,发生了异常时抛出。静态初始化程序是指直接包含于类中的static语句段。 java.lang.IllegalAccessError 违法访问错误。当一个应用试图访问、修改某个类的域(Field)或者调用其方法,但是又违反域或方法的可见性声明,则抛出该异常java.lang.IncompatibleClassChangeError 不兼容的类变化错误。当正在执行的方法所依赖的类定义发生了不兼容的改变时,抛出该异常。一般在修改了应用中的某些类的声明定义而没有对整个应用重新编译而直接运行的情况下,容易引发该错误。 java.lang.InstantiationError 实例化错误。当一个应用试图通过Java的new操作符构造一个抽象类或者接口时抛出该异常. java.lang.InternalError 内部错误。用于指示Java虚拟机发生了内部错误。 java.lang.LinkageError 链接错误。该错误及其所有子类指示某个类依赖于另外一些类,在该类编译之后,被依赖的类改变了其类定义而没有重新编译所有的类,进而引发错误的情况。 java.lang.NoClassDefFoundError 未到类定义错误。当Java虚拟机或者类装载器试图实例化某个类,而不到该类的定义时抛出该错误。 java.lang.NoSuchFieldError 域不存在错误。当应用试图访问或者修改某类的某个域,而该类的定义中没有该域的定义时抛出该错误。 java.lang.NoSuchMethodError 方法不存在错误。当应用试图调用某类的某个方法,而该类的定义中没有该方法的定义时抛出该错误。 java.lang.OutOfMemoryError 内存不足错误。当可用内存不足以让Java虚拟机分配给一个对象时抛出该错误。 java.lang.StackOverflowError 堆栈溢出错误。当一个应用递归调用的层次太深而导致堆栈溢出时抛出该错误。 java.lang.ThreadDeath 线程结束。当调用Thread类的stop方法时抛出该错误,用于指示线程结束。 java.lang.UnknownError 未知错误。用于指示Java虚拟机发生了未知严重错误的情况。 java.lang.UnsatisfiedLinkError 未满足的链接错误。当Java虚拟机未到某个类的声明为native方法的本机语言定义时抛出。 java.lang.UnsupportedClassVersionError 不支持的类版本错误。当Java虚拟机试图从读取某个类文件,但是发现该文件的主、次版本号不被当前Java
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值