我试图查看是否已经存在类似的问题,但找不到,这里是。
我们有一个遗留代码,其中单个BO使用反射对许多DAO进行方法调用。
为了简单起见,我修改了代码。
@Transactional
class EndpointData1DAO implements DAO{
void inserData() {
// insert data 1
}
}
@Transactional
class EndpointData2DAO implements DAO{
void inserData() {
// insert data 2
}
}
class MachoBO {
void handleEverything(String daoName) {
DAO dao = getDAOUsingReflection(daoName);
dao.insertData();
}
}
问题是,需求已更改,因此在EndpointData1DAO上调用insertData()时,还必须调用EndpointData2DAO的insertData。
我可以简单地将EndpointData2DAO添加为EndpointData1DAO的成员,但这严重违反了SRP并使其难看。
所以我写了一个注释@ExecuteAfter(clazz = EndpointData2DAO.class,method =“ insertData”),它获取EndpointData2DAO的一个实例并调用insertData(),
通过使用aop:after执行被注释的类的方法之后,
@Transactional
@ExecuteAfter(clazz=EndpointData2DAO.class, method="insertData")
class EndpointData1DAO implements DAO{
void inserData() {
System.out.println("insert1");
}
}
@Transactional
class EndpointData2DAO implements DAO{
void inserData() {
System.out.println("insert2");
}
}
class MachoBO {
void handleEverything(String daoName) {
DAO dao = getDAOUsingReflection(daoName);
dao.insertData();
}
1个
2
将在调用machoBO.handleEverthing(“ Data1”);时打印出来。
现在我的问题是
EndpointData1DAO和EndpointData2DAO的insertData()是否在同一物理事务中?
换句话说,EndpointData2DAO的insertData()中的运行时异常会回滚EndpointData1DAO的insertData()插入的数据吗?
提前很多谢谢!
最佳答案
我通过实际运行测试找到了答案。
通过执行以下代码,我能够找出当前正在运行的事务的名称。
ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader();
Class tsmClass = contextClassLoader.loadClass("org.springframework.transaction.support.TransactionSynchronizationManager");
String transactionName = (String) tsmClass.getMethod("getCurrentTransactionName", null).invoke(null, null);
System.out.println(transactionName);
并发现当我把
在MachoBO上@Transactional,如下所示,
@Transactional
class MachoBO {
void handleEverything(String daoName) {
DAO dao = getDAOUsingReflection(daoName);
dao.insertData();
}
由于@Transactional具有“方法范围”,
调用
machoBO.handleEverthing("Data1");
时,两个DAO的inserData()在事务名称“ MachoBO.handleEverthing”下执行。
但是,如果MachoBO未使用@Transactional进行注释,
两个DAO的inserData()都不共享相同的方法范围,因此,
在单独的事务(即“ EndpointData1DAO.inserData”和“ EndpointData2DAO.insertData”)下执行。
但是,应该注意(尽管很明显),在使用@Transactional注释DAO并将传播设置为REQUIRES_NEW的情况下,DAO的insertData()在单独的事务中运行。