gpt4 book ai didi

java - Spring Boot集成测试中如何获取JPA上下文?

转载 作者:行者123 更新时间:2023-12-01 19:16:40 24 4
gpt4 key购买 nike

我正在使用 jpa 运行 Springboot 应用程序。我正在尝试建立一个基于 Cucumber 的集成测试。当我在测试中尝试访问存储库时,我收到“org.hibernate.LazyInitializationException”(带有消息“No Session”)。这仅发生在我的集成测试中,但不会发生在实际应用程序中。解决方法是将 @Transactional 放在执行调用的方法上,但如果我在新线程中执行,这将不起作用。

我的第一个问题是:为什么没有 @Transactional 注释它就不起作用?我的第二个问题是:为什么它不能在新线程中使用 @Transactional 注释?

这是我的代码的简化版本:

cucumber 测试:

@RunWith(Cucumber.class)
@CucumberOptions(features = "src/test/resources/some-integration-test.feature")
public class IntegrationTests {}

cucumber 步骤:

@RunWith(SpringRunner.class)
@SpringBootTest(classes = Application.class)
public class IntegrationSteps {

@Autowired
SomeRepo repo;

@When("two updates happen at the same time")
public void twoUpdatesHappenAtTheSameTime() {
ExecutorService executorService = Executors.newFixedThreadPool(2);

executorService.execute(set("Thread 1"));

executorService.execute(set("Thread 2"));

executorService.shutdown();
executorService.awaitTermination(1, TimeUnit.MINUTES);
}

public void set(String someThing) {
Some some = repo.getOne(1234);
repo.setSomeThing(someThing);
repo.save(some);
}
}

以及存储库:

@Repository
public interface SomeRepo extends JpaRepository<Some, Integer> {}

最佳答案

问题似乎出在 getOne() 上。使用getOne(),您只能获得对实体的引用。在尝试访问实体的字段之前,不会执行对数据库的真正调用。通常,当使用 getOne() 时,这些字段会被延迟加载,但由于某种原因(我仍然不清楚),这在 SpringBootTest 中不起作用。

我找到了解决此问题的两种解决方案:

  1. 使用 @Transactional 注释测试。这样您将有一个加载实体的上下文。缺点是您似乎仍然没有获得实体的最新版本。就我而言,我更新了代码中的实体,而在测试代码中,实体中不存在更新(即使更新发生在 getOne() 调用之前)。
  2. 不要使用 getOne(),而是使用 findById()。缺点是 findById() 是急切加载的,但由于我们仅将其用于测试,因此它不会影响应用程序的性能。

关于java - Spring Boot集成测试中如何获取JPA上下文?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59408874/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com