public class TransactionTest {public static void main(String[] args) throws Exception {Connection conn = ConnectionFactory.getInstance().getLocalConnection();int money=0;ResultSet rt=null;try {conn.setAutoCommit(false);Statement st =conn.createStatement();String addMoneySql = "update test_user set money=money+50 where uid=1;";int flag0=st.executeUpdate(addMoneySql);System.out.println(flag0);String reduceMoneySql = "update test_user set money=money-50 where uid=3";int flag1=st.executeUpdate(reduceMoneySql);System.out.println(flag1);String queryMoneySql="select money from test_user where uid=3";rt=st.executeQuery(queryMoneySql);if(rt.next()){money=rt.getInt("mone");//這裡故意將欄位打錯,檢驗java如何執行事務的,原欄位為moneySystem.out.println(money);}if(money<=700){throw new RuntimeException("3號的錢已經不夠了,不要匯款了");//拋出一個異常,在異常處理中復原操作} conn.commit();} catch (Exception e) {e.printStackTrace();//列印出上面拋出來的異常,否則是不會顯示的。if(conn!=null){conn.rollback();}}}}
執行這段代碼。發現結果為:
at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:910)
at com.mysql.jdbc.ResultSet.findColumn(ResultSet.java:987)
at com.mysql.jdbc.ResultSet.getInt(ResultSet.java:2749)
at TransactionTest.main(TransactionTest.java:26)
在異常發生後,catch住異常,執行了復原操作,查看資料庫,資料庫資料並沒有發生變化。但是用update執行的結果來看,update是執行了的。這就說明java處理資料庫的事務的執行策略:先挨句執行(下一句的基本資料是上一句執行完畢的資料,而不是資料庫真實的資料),當出現復原時候,直接通知數據庫復原。資料庫復原以後並不會通知java,所以就會出現列印執行結果與資料庫資料不一致的情況。