JFinal Acton声明事务与手动回滚相结合

    网上已经有很多关于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();
    }
}


你可能感兴趣的:(jFinal)