0

我得到“传递给持久化的分离实体”,但我不明白有问题的对象如何处于分离状态。
首先是一些上下文。

JpaRepository每个用户和角色两个:

@Repository
interface RoleRepository : JpaRepository<UserRole, Long> {
    fun findByName(name: String): UserRole?
}

@Repository
interface BackendUserRepository : JpaRepository<BackendUser, Long> {
    fun findByUserName(name: String): BackendUser?
}

BackendUser实体 - 除了名称 - 有几个不应与此问题相关的字段,外部字段角色来自其基类:

abstract class User(
        @ManyToOne(fetch = FetchType.LAZY, cascade = arrayOf(CascadeType.ALL), optional = false)
        @JoinColumn(referencedColumnName = "name", nullable = false)
        var role: UserRole
) : UserDetails {
    // ...
}

在应用程序启动时,我想确保此 ApplicationRunner 中存在管理员用户:

@Component
class InitialDataApplicationRunner : ApplicationRunner {
    @Autowired
    lateinit var roles: RoleRepository
    @Autowired
    lateinit var users: BackendUserRepository

    override fun run(args: ApplicationArguments) {
        // some other stuff

        createAdminUser()
    }

    @Transactional
    private fun createAdminUser() {
        if (users.findByUserName("admin") == null) {
            val adminRole = roles.findByName("admin")!!
            val adminUser = BackendUser("admin", adminRole
                 // some more data here
            )
            users.save(adminUser)
        }
    }
}

应用程序在启动时抛出异常并关闭:

org.hibernate.PersistentObjectException:分离的实体传递给坚持:my.package.name.user.UserRole

我在这里理解它的方式是,adminRole在从存储库中检索到它之后立即处于分离状态。

  • 为什么是分离的?我曾假设该方法上的 @Transactional 注释将确保所有事情都发生在同一个持久性上下文中,而这不应该发生。
  • 我怎样才能防止这种情况?有没有办法不必删除级联选项
  • 是否有任何关于 spring/jpa 和 hibernate 如何协同工作的好的信息源 - 到目前为止,我发现的任何东西都没有真正帮助我详细理解这些概念,以及存储库如何隐藏你找到的实体管理和会话概念,当试图阅读休眠时。
4

1 回答 1

1

由于您在私有方法上有@Transactional,并且还从同一个类调用了一个,因此没有事务,因此没有用于保存的分离状态(请参阅this)。不确定您是否可以在 run() 或此类上应用 @Transactional。无论如何,也许您应该使用公共 @Transactional 方法创建新的“服务”类,然后从您的类中调用此服务。

于 2020-07-13T11:00:06.083 回答