【问题标题】:Save Many-to-many relation with extra columns Jpa使用额外的列 Jpa 保存多对多关系
【发布时间】:2017-08-24 04:10:53
【问题描述】:

我就是这种情况。我有三个实体:Form、FormComponent 和 FormComponentJoin。 它是与额外列的多对多关系。 我想通过一次调用 FormRepository save() 方法来保存一个表单实体和他的所有孩子,但它不起作用。 这是错误消息: 错误是由以下原因引起的:org.hibernate.TransientPropertyValueException:对象引用了未保存的瞬态实例 - 在刷新之前保存瞬态实例 这些是我的实体:

@Data
@Entity
@Table(name = "form")
@AllArgsConstructor
@NoArgsConstructor
@EqualsAndHashCode
public class Form implements Serializable {

    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(generator = "system-uuid")
    @GenericGenerator(name = "system-uuid", strategy = "uuid")
    private String id;

    private String name;

    @OneToMany(mappedBy = "form", cascade = { CascadeType.PERSIST, CascadeType.MERGE }, orphanRemoval = true)
    @NotFound(action = NotFoundAction.IGNORE)
    @Singular
    private List<FormComponentJoin> formComponentJoin = new ArrayList<>();

}

@Data
@EqualsAndHashCode
@Entity
@Table(name = "form_component")
@AllArgsConstructor(suppressConstructorProperties = true)
@NoArgsConstructor
public class FormComponent implements Serializable {

    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(generator = "system-uuid")
    @GenericGenerator(name = "system-uuid", strategy = "uuid")
    private String id;

    @JsonIgnore
    @OneToMany(mappedBy = "formComponent", cascade = CascadeType.ALL, orphanRemoval = true)
    @NotFound(action = NotFoundAction.IGNORE)
    private List<FormComponentJoin> formComponentJoin = new ArrayList<>();

    @ManyToOne
    @JsonIgnore
    private FormComponent parent;

    @OneToMany(mappedBy = "parent", cascade = CascadeType.ALL)
    @Singular
    private List<FormComponent> children;

    private String value;
}

@Entity
@Table(name = "form_component_join")
@Data
@AllArgsConstructor
@NoArgsConstructor
@EqualsAndHashCode
public class FormComponentJoin implements Serializable {

    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(generator = "system-uuid")
    @GenericGenerator(name = "system-uuid", strategy = "uuid")
    private String id;

    @ManyToOne
    @JoinColumn(name = "form_id", referencedColumnName = "id")
    @NotFound(action = NotFoundAction.IGNORE)
    @JsonIgnore
    private Form form;

    @ManyToOne
    @JoinColumn(name = "form_component_id", referencedColumnName = "id")
    @NotFound(action = NotFoundAction.IGNORE)
    private FormComponent formComponent;

    @Column(name = "position")
    private int pos;

}

@Repository
@Transactional(readOnly = true)
public interface FormRepository extends PagingAndSortingRepository<Form, String> {

}

为了进行测试,我使用 CommandLineRunner:

@Bean
public CommandLineRunner initDatabase(FormRepository formRepository) {
    return (args) -> {

    FormComponent formComponent1 = FormComponent.builder().value("test").build();
    formComponent1.setChildren(null);
    formComponent1.setParent(null);

    FormComponentJoin formComponentJoin1 = FormComponentJoin.builder().pos(1).formComponent(formComponent1).build();

    Form form = Form.builder().name("test").build()).child(formComponentJoin1).build();
    formComponentJoin1.setForm(form);       

    form = formRepository.save(form);   

}

这是完整的堆栈跟踪:

原因:org.hibernate.TransientPropertyValueException:对象引用了未保存的瞬态实例 - 在刷新之前保存瞬态实例:org.fao.backend.entity.FormComponentJoin.formComponent -> org.fao.backend.entity.FormComponent 在 org.hibernate.engine.spi.CascadingActions$8.noCascade(CascadingActions.java:398) 在 org.hibernate.engine.internal.Cascade.cascade(Cascade.java:129) 在 org.hibernate.event.internal.AbstractSaveEventListener.cascadeBeforeSave(AbstractSaveEventListener.java:442) 在 org.hibernate.event.internal.DefaultPersistEventListener.justCascade(DefaultPersistEventListener.java:188) 在 org.hibernate.event.internal.DefaultPersistEventListener.entityIsPersistent(DefaultPersistEventListener.java:181) 在 org.hibernate.event.internal.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:145) 在 org.hibernate.internal.SessionImpl.firePersistOnFlush(SessionImpl.java:838) 在 org.hibernate.internal.SessionImpl.persistOnFlush(SessionImpl.java:831) 在 org.hibernate.engine.spi.CascadingActions$8.cascade(CascadingActions.java:357) 在 org.hibernate.engine.internal.Cascade.cascadeToOne(Cascade.java:350) 在 org.hibernate.engine.internal.Cascade.cascadeAssociation(Cascade.java:293) 在 org.hibernate.engine.internal.Cascade.cascadeProperty(Cascade.java:161) 在 org.hibernate.engine.internal.Cascade.cascadeCollectionElements(Cascade.java:379) 在 org.hibernate.engine.internal.Cascade.cascadeCollection(Cascade.java:319) 在 org.hibernate.engine.internal.Cascade.cascadeAssociation(Cascade.java:296) 在 org.hibernate.engine.internal.Cascade.cascadeProperty(Cascade.java:161) 在 org.hibernate.engine.internal.Cascade.cascade(Cascade.java:118) 在 org.hibernate.event.internal.AbstractFlushingEventListener.cascadeOnFlush(AbstractFlushingEventListener.java:167) 在 org.hibernate.event.internal.AbstractFlushingEventListener.prepareEntityFlushes(AbstractFlushingEventListener.java:158) 在 org.hibernate.event.internal.AbstractFlushingEventListener.flushEverythingToExecutions(AbstractFlushingEventListener.java:91) 在 org.hibernate.event.internal.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:55) 在 org.hibernate.internal.SessionImpl.flush(SessionImpl.java:1258) 在 org.hibernate.internal.SessionImpl.managedFlush(SessionImpl.java:425) 在 org.hibernate.engine.transaction.internal.jdbc.JdbcTransaction.beforeTransactionCommit(JdbcTransaction.java:101) 在 org.hibernate.engine.transaction.spi.AbstractTransactionImpl.commit(AbstractTransactionImpl.java:177) 在 org.hibernate.jpa.internal.TransactionImpl.commit(TransactionImpl.java:77) ...省略了58个常用框架

【问题讨论】:

    标签: hibernate spring-boot spring-data-jpa lombok postgresql-9.6


    【解决方案1】:

    最终我了解到错误是 FormComponentJoin 实体中的映射。

    这是新的映射:

    @ManyToOne(cascade = { CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE })
    @JoinColumn(name = "form_id") // , referencedColumnName = "id"
    @NotFound(action = NotFoundAction.IGNORE)
    @JsonIgnore
    private Form form;
    
    @ManyToOne(cascade = { CascadeType.PERSIST, CascadeType.MERGE })
    @JoinColumn(name = "form_component_id") // , referencedColumnName = "id"
    @NotFound(action = NotFoundAction.IGNORE)
    private FormComponent formComponent;
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2020-03-03
      相关资源
      最近更新 更多