一、結論
Spring的事務管理默認只對出現運行期異常(java.lang.RuntimeException及其子類)進行回滾。
如果一個方法拋出Exception或者Checked異常,Spring事務管理默認不進行回滾。
關於異常的分類一下詳細介紹:
1、基本概念
看java的異常結構圖
Throwable是所有異常的根,java.lang.Throwable
Error是錯誤,java.lang.Error
Exception是異常,java.lang.Exception
2、Exception
一般分為Checked異常和Runtime異常,所有RuntimeException類及其子類的實例被稱為Runtime異常,不屬於該范疇的異常則被稱為CheckedException。
①Checked異常
只有java語言提供了Checked異常,Java認為Checked異常都是可以被處理的異常,所以Java程序必須顯示處理Checked異常。如果程序沒有處理Checked異常,該程序在編譯時就會發生錯誤無法編譯。這體現了Java的設計哲學:沒有完善錯誤處理的代碼根本沒有機會被執行。對Checked異常處理方法有兩種
(1) 當前方法知道如何處理該異常,則用try...catch塊來處理該異常。
(2) 當前方法不知道如何處理,則在定義該方法是聲明拋出該異常。
代碼如下:
package cn.xy.test;
import java.io.IOException;
/**
* Checked異常測試方法
* @author xy
*
*/
public class CheckedExceptionMethods
{
// 總異常類,既有checkedException又有RuntimeException,所以其中的checkedException必須處理
public void method1() throws Exception
{
System.out.println("我是拋出異常總類的方法");
}
// 捕獲並處理這個異常
public void testMethod1_01()
{
try
{
method1();
}
catch (Exception e)
{
e.printStackTrace();
}
}
// 把異常傳遞下去
public void testMethod1_02() throws Exception
{
method1();
}
public void testMethod1_03() throws Exception
{
throw new Exception();
}
public void testMethod1_04()
{
try
{
throw new Exception();
}
catch (Exception e)
{
e.printStackTrace();
}
}
// checkedException典型代表IOException
public void method2() throws IOException
{
System.out.println("我是拋出IO異常的方法");
}
public void testMethod2_01()
{
try
{
method2();
}
catch (Exception e)
{
e.printStackTrace();
}
}
public void testMethod2_02() throws Exception
{
method2();
}
}
我們比較熟悉的Checked異常有
Java.lang.ClassNotFoundException
Java.lang.NoSuchMetodException
java.io.IOException
②RuntimeException
Runtime如除數是0和數組下標越界等,其產生頻繁,處理麻煩,若顯示申明或者捕獲將會對程序的可讀性和運行效率影響很大。所以由系統自動檢測並將它們交給缺省的異常處理程序。當然如果你有處理要求也可以顯示捕獲它們。
代碼如下:
package cn.xy.test;
/**
* 運行時異常測試方法
* @author xy
*
*/
public class RuntimeExcetionMethods
{
public void method3() throws RuntimeException
{
System.out.println("我是拋出運行時異常的方法");
}
public void testMethod3_01()
{
method3();
}
public void testMethod1_02()
{
throw new RuntimeException();
}
}
我們比較熟悉的RumtimeException類的子類有
Java.lang.ArithmeticException
Java.lang.ArrayStoreExcetpion
Java.lang.ClassCastException
Java.lang.IndexOutOfBoundsException
Java.lang.NullPointerException
3、Error
當程序發生不可控的錯誤時,通常做法是通知用戶並中止程序的執行。與異常不同的是Error及其子類的對象不應被拋出。
Error是throwable的子類,代表編譯時間和系統錯誤,用於指示合理的應用程序不應該試圖捕獲的嚴重問題。
Error由Java虛擬機生成並拋出,包括動態鏈接失敗,虛擬機錯誤等。程序對其不做處理。
二、改變默認方式
在@Transaction注解中定義noRollbackFor和RollbackFor指定某種異常是否回滾。
@Transaction(noRollbackFor=RuntimeException.class)
@Transaction(RollbackFor=Exception.class)
這樣就改變了默認的事務處理方式。
三、啟示
這就要求我們在自定義異常的時候,讓自定義的異常繼承自RuntimeException,這樣拋出的時候才會被Spring默認的事務處理准確處理。