繁体   English   中英

Spring 引导自动提交手动事务,无需调用存储库保存

[英]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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM