Java JUnit:@Rule';s重写的apply方法未在测试下执行
我创建了一个JUnit规则,以便在发生任何异常时回滚javax.persistence事务(否则,由于事务处于不一致的状态,所有进一步的测试都将失败)。问题是当测试开始时,我的规则永远不会执行,严格地说:Java JUnit:@Rule';s重写的apply方法未在测试下执行,java,junit,rule,Java,Junit,Rule,我创建了一个JUnit规则,以便在发生任何异常时回滚javax.persistence事务(否则,由于事务处于不一致的状态,所有进一步的测试都将失败)。问题是当测试开始时,我的规则永远不会执行,严格地说:apply方法永远不会执行。当我将@Rule声明放入具体类并在测试中初始化transactionRule时,它甚至不起作用。下面是它的样子: 规则 public class TransactionRule implements TestRule { private EntityManag
apply
方法永远不会执行。当我将@Rule声明放入具体类并在测试中初始化transactionRule时,它甚至不起作用。下面是它的样子:
规则
public class TransactionRule implements TestRule
{
private EntityManager entityManager;
public TransactionRule(EntityManager entityManager)
{
this.entityManager = entityManager;
}
@Override
public Statement apply(Statement base, Description description)
{
return new TransactionStatement(base);
}
public class TransactionStatement extends Statement
{
private final Statement runningTest;
public TransactionStatement(Statement runningTest)
{
this.runningTest = runningTest;
}
@Override
public void evaluate() throws Throwable
{
try
{
runningTest.evaluate();
}
catch (Exception e)
{
if (entityManager.getTransaction().isActive())
{
entityManager.getTransaction().rollback();
}
}
}
}
}
public abstract class AbstractIntegrationTest
{
//more members vars
@Rule
public TransactionRule transactionRule;
@BeforeClass
public static void setUpBeforeClass()
{
loadProperties();
entityManagerFactory = Persistence.createEntityManagerFactory("MyCuisinePersistenceTestUnit", connectionProperties);
entityManager = entityManagerFactory.createEntityManager();
}
@Before
public void setUp()
{
transactionRule = new TransactionRule(entityManager);
entityManager.clear();
}
//more code
}
使用规则的抽象类
public class TransactionRule implements TestRule
{
private EntityManager entityManager;
public TransactionRule(EntityManager entityManager)
{
this.entityManager = entityManager;
}
@Override
public Statement apply(Statement base, Description description)
{
return new TransactionStatement(base);
}
public class TransactionStatement extends Statement
{
private final Statement runningTest;
public TransactionStatement(Statement runningTest)
{
this.runningTest = runningTest;
}
@Override
public void evaluate() throws Throwable
{
try
{
runningTest.evaluate();
}
catch (Exception e)
{
if (entityManager.getTransaction().isActive())
{
entityManager.getTransaction().rollback();
}
}
}
}
}
public abstract class AbstractIntegrationTest
{
//more members vars
@Rule
public TransactionRule transactionRule;
@BeforeClass
public static void setUpBeforeClass()
{
loadProperties();
entityManagerFactory = Persistence.createEntityManagerFactory("MyCuisinePersistenceTestUnit", connectionProperties);
entityManager = entityManagerFactory.createEntityManager();
}
@Before
public void setUp()
{
transactionRule = new TransactionRule(entityManager);
entityManager.clear();
}
//more code
}
有缺陷测试的测试类别
public class RecipePersistenceITest extends AbstractIntegrationTest
{
//more tests
@Test
public void persistenceOfRecipeWithUserCategorySuccessful()
{
//Test that fails
}
}
有什么想法吗?测试规则是在使用
@before
注释的方法之前调用的,因此尝试在@before
中分配规则将无效(尽管我希望它会引发异常)
相反,在定义上指定规则(并使字段成为最终字段),并在之前的@中执行任何其他配置(如有必要)
请注意,每个测试方法都在测试类的一个新实例中执行,因此将规则定义为最终字段没有问题。我不完全确定,但我认为在
之前的@中创建规则为时已晚。谢谢。顺便说一句,为了不让任何其他测试在一个测试失败时崩溃,我只需要在@Before中创建一个新的entityManager,因此最终,甚至不需要该规则。但是问题和答案仍然对我有帮助,因为在公司里,我也遇到了同样的问题,我想知道为什么我的规则没有被执行。@Bevor每个测试都在它们自己的类实例中运行,您应该利用这个属性。例如,您可以在规则中创建和销毁实体管理器。