网上已经有很多关于jfinal事务的文章或者贴子,但是却没有详细一点,或者说实际一点的。
我们在开发中,使用事务的原因,是因为有一系列的数据库操作需要进行,但因为某些原因时,希望系统能够回滚。
我不清楚别人是如何使用jFinal事务的,在这里只想谈谈我的做法。同时感谢Jfinal作者的耐心解答。
用例:一个Controller方法,通过它执行一系列的操作。当然它其中调用了很多方法。
在Action声明上添加声明事务
@Before(Tx.class)
public void saveData(){
// 省略
}
这样,saveData就可以支持事务了,不管saveData中怎样处理,或者调用多少方法,都没问题。
但是,在开发过程出现了一些问题。saveData中,或者所调用的方法中,如果添加了try catch,异常不能够截断它,必须继续向上抛。
因为Tx声明事务,是通过Tx拦截器进行的,其中Tx,通过异常对事务进行回滚。可以打开代码,如下所示,它是Tx.java中的一段。
try {
conn = config.getConnection();
autoCommit = conn.getAutoCommit();
config.setThreadLocalConnection(conn);
conn.setTransactionIsolation(getTransactionLevel(config));// conn.setTransactionIsolation(transactionLevel);
conn.setAutoCommit(false);
ai.invoke();
conn.commit();
} catch (NestedTransactionHelpException e) {
if (conn != null) try {conn.rollback();} catch (Exception e1) {e1.printStackTrace();}
} catch (Throwable t) {
if (conn != null) try {conn.rollback();} catch (Exception e1) {e1.printStackTrace();}
throw new ActiveRecordException(t);
}
那么当我们使用声明式Tx事务时,想回滚只能抛出异常了。
@Before(Tx.class)
public void saveData(){
// 省略
if(xxxx){
throw new Exception("xxxxx");
}
}
这样,saveData中,调用若干方法进行Db.save,Db.update,Db.delete都没问题。但是要注意事务级别
ActiveRecordPlugin arp = new ActiveRecordPlugin(druidPlugin);
arp.setTransactionLevel(4);
Db.save 相当于将数据库中的数据先读出来,然后再用 sql 写回去
Db.update 这类属于直接操作数据库
Db.update只需要级2就可以,但是Db.save需要级别4。
那么,除了抛出异常,在继续使用Tx声明事务的情况下,有没有不抛出进行回滚的呢?
可以这样写:
@Before(Tx.class)
public void saveData(){
// 省略
if(xxxx){
// message
// 略
// 手动回滚
DbKit.getConfig().getConnection().rollback();
}
}