Spring JpaRepository - 分离和附加实体

Posted

技术标签:

【中文标题】Spring JpaRepository - 分离和附加实体【英文标题】:Spring JpaRepository - Detach and Attach entity 【发布时间】:2015-01-03 21:38:51 【问题描述】:

我在 jpa 上使用 spring boot 和 hibernate。我正在使用 JpaRepository 接口来实现我的存储库。与以下 UserRepository 一样

public interface UserRepository extends JpaRepository<User, Long> 

我想实现以下目标

    加载用户实体。 更改实体对象的状态,例如user.setName("foo") 执行外部系统 Web 服务调用。将调用结果保存在数据库中 仅在成功响应此 Web 服务调用时,将用户的新状态保存在存储库中。

上述所有步骤都没有发生在一个事务中,即外部服务调用不在事务中。

当我通过其存储库将我的 Web 服务结果保存到 DB 中时,我在用户实体中的更改也会被保存。根据我的理解,这是由于在第 3 步刷新了底层持久性上下文。经过一番 google,我认为我可以实现我的目的,如果我可以在第 1 步分离我的用户实体并在第 4 步重新附加它。 请确认我的理解是否正确以及如何实现? JpaRepository 接口中没有分离实体的方法。

以下是说明的代码

public void updateUser(int id, String name, int changeReqId)
    User mUser = userRepository.findOne(id); //1
    mUser.setName(name); //2

    ChangeRequest cr = changeRequestRepository.findOne(changeReqId);
    ChangeResponse rs = userWebService.updateDetails(mUser); //3

    if(rs.isAccepted())
        userRepository.saveAndFlush(mUser); //4
    

    cr.setResponseCode(rs.getCode());
    changeRequestRepository.saveAndFlush(cr); //this call also saves the changes at step 2

谢谢

【问题讨论】:

您应该为步骤 1 到 4 发布您当前的代码(以澄清) 是否可以按以下顺序拨打电话:1,3,4,2? 【参考方案1】:

使用@Predrag Maric 建议的自定义实现显然是这个问题的正确答案。但是,我发现在服务层进行分离要好得多,因为通常它知道实体是否应该分离。

只需将其与服务中的@PersistenceContext 连接即可。

@Service
class ConsumerServiceImpl 

    @PersistenceContext
    private EntityManager entityManager
...

    entityManager.detach(en)

【讨论】:

【参考方案2】:

如果您使用的是 JPA 2.0,则可以使用 EntityManager#detach() 将单个实体从持久性上下文中分离出来。此外,Hibernate 有一个 Session#evict() 用于相同目的。

由于JpaRepository 本身不提供此功能,您可以add a custom implementation 给它,类似这样

public interface UserRepositoryCustom 
    ...
   void detachUser(User u);
    ...


public interface UserRepository extends JpaRepository<User, Long>, UserRepositoryCustom 
    ...


@Repository
public class UserRepositoryCustomImpl implements UserRepositoryCustom 
    ...
    @PersistenceContext
    private EntityManager entityManager;

    @Override
    public void detachUser(User u) 
        entityManager.detach(u);
    
    ...

我没有尝试过这段代码,但你应该可以让它工作。您甚至可以尝试使用@PersistenceContext 在您的服务类(updateUser() 所在的位置)中保留EntityManager,并避免向存储库添加自定义实现的麻烦。

【讨论】:

谢谢,这种方法在我的 Spring Boot 应用程序中有效。 (冒昧地将@Repository 添加到示例代码中的实现中;否则自定义存储库无法自动装配。) CrudRepository 可以使用相同的方法吗? @DBS 没试过,应该是一样的【参考方案3】:

entityManager.clear() 将断开所有 JPA 对象,因此在所有情况下这可能不是一个合适的解决方案,如果您确实计划保持连接的其他对象。

清除

/**
 * Clear the persistence context, causing all managed
 * entities to become detached. Changes made to entities that
 * have not been flushed to the database will not be
 * persisted.
 */
public void clear();

entityManager.detach(entity);从持久化上下文中移除给定的实体

分离

/**
 * Remove the given entity from the persistence context, causing
 * a managed entity to become detached.  Unflushed changes made
 * to the entity if any (including removal of the entity),
 * will not be synchronized to the database.  Entities which
 * previously referenced the detached entity will continue to
 * reference it.
 * @param entity  entity instance
 * @throws IllegalArgumentException if the instance is not an
 *         entity
 * @since Java Persistence 2.0
 */
public void detach(Object entity);

【讨论】:

以上是关于Spring JpaRepository - 分离和附加实体的主要内容,如果未能解决你的问题,请参考以下文章

spring使用 hibernate jpa JpaRepository

Spring JPA no @Transnational 保存 JpaRepository

Spring Data JPA - JpaRepository 中的自定义排序

将 JpaRepository 与 Spring 数据和 Hibernate 一起使用

如何在 Spring 中将 OrderBy 与 JPARepository 一起使用

在 spring jparepository 中加入多个表