我们遇到了一个生产事件导致一堆线程死锁,服务器停止工作。为了尝试和调查,我测试了一些具有不同弹簧事务传播的东西,如果我没有弄错,如果根本没有现有的事务,则REQUIRES_NEW传播将启动两个连接。它是否正确??我尝试使用谷歌搜索,但没有找到关于此的信息。
我做了一个测试。这是一个示例类:
package test;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Service
public class TheService {
@Transactional(propagation=Propagation.REQUIRES_NEW)
public void doSomething() {
System.out.println("Here I am doing something.");
}
}
这是我做的单元测试:
package test;
import javax.annotation.Resource;
import org.junit.Test;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.AbstractTransactionalJUnit4SpringContextTests;
@ContextConfiguration(locations = {"classpath:test.xml"})
public class TheServiceTest extends AbstractTransactionalJUnit4SpringContextTests {
@Resource
private TheService theService;
@Test
public void test() {
theService.doSomething();
}
}
最后但并非最不重要的是,这是我的测试xml:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.1.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.1.xsd"
default-autowire="byName">
<context:component-scan base-package="test" />
<tx:annotation-driven transaction-manager="transactionManager" />
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="entityManagerFactory" ref="operator.entityManagerFactory" />
</bean>
<bean id="operator.entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="persistenceUnitName" value="operatorPersistenceUnit" />
<property name="dataSource" ref="operator.dataSource" />
<property name="jpaVendorAdapter">
<bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
<property name="showSql" value="false" />
<property name="generateDdl" value="true" />
<property name="databasePlatform" value="org.hibernate.dialect.H2Dialect" />
</bean>
</property>
</bean>
<bean id="operator.dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="org.h2.Driver" />
<property name="url" value="jdbc:h2:mem:operator" />
<property name="username" value="sa" />
<property name="password" value="" />
<property name="maxActive" value="1" /> <!-- NOTE -->
</bean>
</beans>
我想要REQUIRES_NEW用于方法的原因是因为不从它获取任何脏读取是至关重要的,并且它可以在另一个事务中和外部执行。
如果我将maxActive属性保持为1,则此测试将死锁并且永远不会打印任何内容。但是,如果我将其更改为2,则测试将通过。
这是一个令人担忧的原因是,即使我将maxActive设置为更高的值,并且有足够的线程等待执行此方法,它们最终都会占用一个连接并等待第二个连接。
我做错了吗?我误解了什么吗?
我感谢任何帮助!谢谢!
答案 0 :(得分:2)
它与propagation=REQUIRES_NEW
无关,默认情况下不会打开2个连接。问题是您正在扩展AbstractTransactionalJUnit4SpringContextTests
。
正如您的测试用例扩展AbstractTransactionalJUnit4SpringContextTests
,正如您所看到的那样@Transactional
。此测试的交易由TransactionalTestExecutionListener
管理。
所以当你开始测试时,在执行测试方法之前,测试框架会启动一个事务。接下来,您调用由于使用@Transactional(propagation=REQUIRES_NEW)
进行注释而启动另一个事务的服务。
修复非常简单,不要AbstractTransactionalJUnit4SpringContextTests
延伸,只需使用@RunWith(SpringRunner.class)
为您的课程添加注释。
@RunWith(SpringRunner.class)
@ContextConfiguration(locations = {"classpath:test.xml"})
public class TheServiceTest { ... }