java – Cascading save不能与Hibernate 4和@OneToMany一起使用

我一直遇到Hibernate 4的问题,这在Hibernate 3中没有发生.我正在测试一个基础dao类,它具有保存,查询等通用方法.(不是它不是功课).我正在使用嵌入式derby连接与hibernate.hbm2ddl.auto = create-drop进行测试.

编辑:修复cascadeType.还是一样的行为.即使我从CascadeType枚举中添加了所有可用的级联类型,它仍然会失败,除非我放入CascadeType.ALL.不确定你有什么不能具体列举的.

在我的测试中,我有以下映射类:

@Entity
@Table(name="FOO_CHILD")
public class FooChild {
    @Id
    @Column(name="FOO_ID")
    private Long id;

    @Column(name="NAME")
    private String name;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

@Entity
@Table(name="FOO_PARENT")
public class FooParent {
    @Id
    @Column(name="FOO_PARENT_ID")
    private Long id;

    @Column(name="NAME")
    private String name;

    @Fetch(FetchMode.SELECT)
    @Type(type="long")
    @OneToMany(cascade={CascadeType.PERSIST, CascadeType.MERGE})
    @JoinTable(name="FOO_PARENT_FOO",
            joinColumns={@JoinColumn(referencedColumnName="FOO_PARENT_ID")},
            inverseJoinColumns={@JoinColumn(referencedColumnName="FOO_ID")})
    private List<Foo> fooChild;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public List<Foo> getFooChild() {
        return fooChild;
    }

    public void setFooChild(List<Foo> fooChild) {
        this.fooChild = fooChild;
    }
}

然后我尝试保存:

FooChild fooChild = new FooChild();
fooChild.setId(1L);
fooChild.setName("fooChild");

FooParent fooParent = new FooParent();
fooParent.setId(1L);
fooParent.setName("fooParent");
fooParent.setFooChild(new ArrayList<FooChild>(Arrays.asList(fooChild)));

session.save(fooParent);
session.flush();
session.clear();

我得到以下痕迹:

org.hibernate.exception.ConstraintViolationException: INSERT on table 'FOO_PARENT_FOO' caused a violation of foreign key constraint 'FKF18290CAECA19CB8' for key (1).  The statement has been rolled back.
    at org.hibernate.exception.internal.SQLExceptionTypeDelegate.convert(SQLExceptionTypeDelegate.java:74)
...

我注意到以下sql正在执行:

Hibernate: 
    insert 
    into
        foo_parent
        (name, foo_parent_id) 
    values
        (?, ?, ?, ?)
Hibernate: 
    insert 
    into
        foo_parent_foo
        (foo_parent, foo_child) 
    values
        (?, ?)

它似乎是在将保存级联到FooChild对象之前尝试插入到集合中.

如果我改变FooParent有这个:

@OneToMany(cascade={CascadeType.ALL})

问题消失了.

正如我所说,这适用于hibernate 3,所以我不确定它为什么表现不同.

最佳答案 save(),persist()和merge()是三种不同的操作.如果要级联save()操作,则需要使用Hibernate注释

@Cascade(CascadeType.SAVE_UPDATE)

要么

cascade = CascadeType.ALL

,其中还包括Hibernate专有操作.

点赞