![](/img/trans.png)
[英]Spring Boot + JPA + Hibernate does not commit when repository.save
[英]Spring Boot auto commit manuell transaction without call save on repository
Spring 引导尝试在不调用存储库中的保存方法的情况下为实体 personTransaction1 自动提交手动事务中的更改。 personTransaction1 中的更改已提交。 方法manuellTransaction抛出 org.springframework.orm.ObjectOptimisticLockingFailureException。 在方法transactionByAnnotation中具有基于注释的事务处理的相同代码按预期工作,并且没有提交变量 personTransaction1 的更改。 spring 尝试提交的原因是什么?
@SpringBootApplication
@EnableJpaAuditing
class DbOptimiticLockingApplication : CommandLineRunner {
@Autowired
private lateinit var personRepository: PersonRepository
@Autowired
private lateinit var platformTransactionManager: PlatformTransactionManager
override fun run(vararg args: String?) {
executeInNewTransaction {
personRepository.deleteAll()
}
val person = createPerson()
transactionByAnnotation(person.id)
manuellTransaction(person.id)
}
@Transactional
fun createPerson(): Person {
val person = Person()
person.name = "Max"
return personRepository.save(person)
}
@Transactional(Transactional.TxType.REQUIRES_NEW)
fun transactionByAnnotation(id: Long) {
var personTransaction1 = personRepository.findById(id).get()
// don't trigger commit
personTransaction1.name = "Tom"
transaction11ByAnnotation(personTransaction1.id)
transaction12ByAnnotation(personTransaction1.id)
}
@Transactional(Transactional.TxType.REQUIRES_NEW)
fun transaction11ByAnnotation(id: Long) {
businessLogic1(id)
}
@Transactional(Transactional.TxType.REQUIRES_NEW)
fun transaction12ByAnnotation(id: Long) {
businessLogic2(id)
}
fun manuellTransaction(id: Long) {
executeInNewTransaction {
var personTransaction1 = personRepository.findById(id).get()
// trigger commit
personTransaction1.name = "Tom"
manuellTransaction11(personTransaction1.id)
manuellTransaction12(personTransaction1.id)
}
}
fun manuellTransaction11(id: Long) {
executeInNewTransaction {
businessLogic1(id)
}
}
fun manuellTransaction12(id: Long) {
executeInNewTransaction {
businessLogic2(id)
}
}
private fun businessLogic1(id: Long) {
val person = personRepository.findById(id).get()
person.name = "Martin"
personRepository.saveAndFlush(person)
}
private fun businessLogic2(id: Long) {
val person = personRepository.findById(id).get()
person.name = "Joe"
personRepository.saveAndFlush(person)
}
private fun <T> executeInNewTransaction(action: () -> T): T {
val transactionTemplate = TransactionTemplate(platformTransactionManager)
transactionTemplate.propagationBehavior = TransactionDefinition.PROPAGATION_REQUIRES_NEW
return try {
transactionTemplate.execute {
action()
}!!
} catch (e: Exception) {
throw e
}
}
}
@Entity
@EntityListeners
class Person {
@Id
@GeneratedValue
var id: Long = 0L
@Column
var name: String = ""
@Version
var version: Long = 0L
}
interface PersonRepository : JpaRepository<Person, Long>
fun main(args: Array<String>) {
runApplication<DbOptimiticLockingApplication>(*args)
}
spring 尝试提交的原因是什么?
当从数据库中读取一个实体时,对于 JPA 层来说,它变成了一个persistent
的或以其他方式称为managed
实体。
ORM 供应商观察到persistent/managed
state 上的实体,并且对它们所做的任何更改都会自动传递到数据库层。 发生这种情况的前提是,实体被视为persistent/managed
的方法完成且没有任何错误,并且该方法属于JPA 事务!
对于您描述的以下引用,不,它没有按预期工作,您观察到的行为和您期望发生的事情只是巧合。
transactionByAnnotation 按预期工作,没有对变量 personTransaction1 进行任何更改
@Transactional(Transactional.TxType.REQUIRES_NEW)
fun transactionByAnnotation(id: Long) {
var personTransaction1 = personRepository.findById(id).get()
@Transactional(Transactional.TxType.REQUIRES_NEW)
-> 在您在此处提到的这种情况下被完全忽略,因为该方法的调用来自方法run
的同一实例,因此根本不考虑注释。 更多信息在一个旧的 SO 问题中
因此,由于此方法在没有任何错误的情况下完成,但不属于任何JPA 事务,因此对托管实体所做的更改不会自动传递到数据库中。
如果它属于 JPA 事务,并且@Transactinal
没有因为自调用而被忽略,它将保持数据库中的更改。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.