gpt4 book ai didi

Spring Boot + Spring Data JPA + 事务无法正常工作

转载 作者:IT老高 更新时间:2023-10-28 13:50:31 24 4
gpt4 key购买 nike

我使用 1.2.0 版本和 spring-boot-starter-data-jpa 创建了一个 Spring Boot 应用程序,并且我正在使用 MySQL。我已在 application.properties 文件中正确配置了我的 MySQL 属性。

我有一个简单的 JPA 实体、Spring Data JPA 存储库和一个服务,如下所示:

@Entity
class Person
{
@Id @GeneratedValue(strategy=GenerationType.AUTO)
private Integer id;
private String name;
//setters & getters
}

@Repository
public interface PersonRepository extends JpaRepository<Person, Integer>{

}


import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service
@Transactional
class PersonService
{
@Autowired PersonRepository personRepository;

@Transactional
void save(List<Person> persons){
for (Person person : persons) {
if("xxx".equals(person.getName())){
throw new RuntimeException("boooom!!!");
}
personRepository.save(person);
}
}
}

我有以下 JUnit 测试:

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.SpringApplicationConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = Application.class)
public class ApplicationTests {

@Autowired
PersonService personService;

@Test
public void test_logging() {
List<Person> persons = new ArrayList<Person>();
persons.add(new Person(null,"abcd"));
persons.add(new Person(null,"xyz"));
persons.add(new Person(null,"xxx"));
persons.add(new Person(null,"pqr"));

personService.save(persons);
}

}

这里的期望是它不应该在 PERSON 表中插入任何记录,因为它会在插入 3rd person 对象时抛出异常。但它并没有回滚,前 2 条记录被插入并提交。

然后我想到了快速尝试使用 JPA EntityManager。

@PersistenceContext
private EntityManager em;

em.save(person);

然后我得到 javax.persistence.TransactionRequiredException: No transactional EntityManager available 异常。

谷歌搜索了一段时间后,我遇到了这个 JIRA 线程 https://jira.spring.io/browse/SPR-11923同一个话题。

然后我将 Spring Boot 版本更新为 1.1.2 以获取早于 4.0.6 的 Spring 版本。

然后 em.save(person) 按预期工作并且 Transaction 工作正常(意味着当 RuntimeException 发生时它正在回滚所有数据库插入)。

但即使使用 Spring 4.0.5 + Spring Data JPA 1.6.0 版本,当使用 personRepository.save(person) 而不是 em.persist(person)

似乎 Spring Data JPA 存储库正在提交事务。

我错过了什么?如何使服务级别的@Transactional 注释起作用?

PS:

Maven pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<groupId>com.sivalabs</groupId>
<artifactId>springboot-data-jpa</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>jar</packaging>

<name>springboot-data-jpa</name>
<description>Spring Boot Hello World</description>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.2.0.RELEASE</version>
<relativePath />
</parent>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<start-class>com.sivalabs.springboot.Application</start-class>
<java.version>1.7</java.version>
</properties>

<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>

<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
</dependencies>
</project>

Application.java

@EnableAutoConfiguration
@Configuration
@ComponentScan
public class Application {

public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}

最佳答案

将事务注释更改为 @Transactional(rollbackFor=Exception.class)

关于Spring Boot + Spring Data JPA + 事务无法正常工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28606518/

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