【问题标题】:Hibernate: Child id is null after merging parent休眠:合并父级后子级ID为空
【发布时间】:2017-01-29 20:58:34
【问题描述】:

我已经在 stackoverflow 上提出了类似的问题,但我无法找到正确的答案,所以如果可以的话,我会发布这个问题:

我在 Spring Boot 应用程序中有一个典型的一对多父子实体。因此,我创建了一个新子项,将其添加到父项,然后合并/刷新/清除实体管理器。子实体以新 ID 正确保存在数据库中,但父对象下的子对象 ID 为零。我错过了什么会强制刷新 ID?

创建新子代的代码位于最初获取父代的同一控制器中,因此它使用相同的 DAO 和相同的实体管理器。

这是我的代码(实体主要由 JPA 生成):

家长:

@Entity
public class Parent implements Serializable {
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    private int id;

    @OneToMany(mappedBy="parent", fetch=FetchType.EAGER, cascade={CascadeType.MERGE})
    private List<Child> children;

    public List<Child> getChildren() {
        return this.Children;
    }

    public void setChildren(List<Child> children) {
        this.children = children;
    }

    public Child addChild(Child child) {
        getChildren().add(child);
        child.setParent(this);
        return child;
    }
}

孩子:

@Entity
public class Child implements Serializable {
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    private int id;

    @ManyToOne(cascade={CascadeType.MERGE})
    @JoinColumn(name="parent_id")
    private Parent parent;

    public Parent getParent() {
        return this.parent;
    }

    public void setParent(Parent parent) {
        this.parent = parent;
    }
}

道:

@Repository
public class ParentDao {
    @PersistenceContext
    private EntityManager entityManager;

    @Transactional
    public void update(Parent parent) {
        entityManager.merge(parent);
        entityManager.flush();
        entityManager.clear();
    }
}

控制器:

@Autowired
private ParentDao parentDao;

. . . createChild() {
    Child child = new Child();
    // Set the link on both sides of the relationship
    parent.addChild(child);
    parentDao.update(parent);
    // --> child.getId() == 0
}

【问题讨论】:

    标签: java hibernate jpa


    【解决方案1】:

    merge() 在新实体(在您的情况下为子实体)上调用时的工作方式,它执行persist() 调用,但有一点不同。

    它只创建传递实体的副本并在副本上调用persist()。

    因此,您的 Parent 实体看不到 id 的变化,因为它持有 Child 的“旧”实例,该实例未持久化并最终使用新的 id 更新。

    希望能解决问题。

    更新

    要获得新状态,您需要:

    1) 更改事务存储库方法以返回 merge() 值:

    @Transactional
        public Parent update(Parent parent) {
            Parent newParent = entityManager.merge(parent);
            entityManager.flush();
            entityManager.clear();
            return newParent
        }
    

    2) 更改您的服务:

    @Autowired
    private ParentDao parentDao;
    
    . . . createChild() {
        Child child = new Child();
        // Set the link on both sides of the relationship
        parent.addChild(child);
        Parent newParent = parentDao.update(parent);
        // --> newParent.getChildren(0).getId() == 0
    }
    

    【讨论】:

    • 我的印象是,flush() 会刷新对象并将其向下级联,但我想它不会......那么答案是什么,我怎样才能刷新新的加了孩子?我也需要flush()它们吗?
    • 谢谢你,maciej。所以,虽然这似乎可行,但我仍然有一个问题 - 休眠中是否没有选项来刷新原始父对象?我想从根本上说,我想将对象管理卸载到 orm,如果我必须付出如此多(容易出错)的努力来管理它们,这会违背目的,不是吗?我会想象很多(如果不是大多数)用户会想要同样的东西 - 只需将我的对象及其新子对象存储在数据库中,然后刷新它。如果我的父对象引用分散在整个项目中,我将如何持续更新它们?
    • 我上面评论中的第二个问题是修辞 - 我只是认为 ORM 强迫我更新对我的对象的所有引用只是为了获取新创建的孩子的 ID 是没有意义的.另外,在我的情况下,Parent 是我的 User 对象,即通过应用程序进行身份验证的对象。每次我添加一个孩子时都替换这个对象让我有点紧张和不安......
    • 使用我的方法,您不必更新任何内容。您只需要获取在 merge() 操作后返回的新实例。
    • 父级 newParent = entityManager.merge(parent);
    猜你喜欢
    • 2016-11-17
    • 1970-01-01
    • 1970-01-01
    • 2023-04-06
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2010-12-31
    相关资源
    最近更新 更多