Как сохранить сущность с двумя двунаправленными зависимостями «многие-к-одному» в Hibernate? - PullRequest
0 голосов
/ 25 февраля 2019

У меня есть следующие реляционные сущности в базе данных:

Table A <-- (One-to-Many) --> Table B
Table B <-- (One-to-Many) --> Table C
Table B <-- (Many-to-One) --> Table J
Table C <-- (Many-to-One) --> Table J

Таблица B содержит внешние ключи для таблиц A и J. Таблица C содержит внешние ключи для таблиц B и J.

ПримечаниеКак в таблицах B, так и в C содержится ссылка на внешний ключ (TABLE_J_ID), ссылающаяся на таблицу J.

Как вы можете видеть, таблица B и C имеют отношение «многие к одному» с таблицей J. Таким образом, Iсоздали следующие двунаправленные объекты:

@Entity
@Table(name="TableA")
public class TableA {
  @Id
  // Generates via a sequence object
  private Integer id;

  @OneToMany(mapped="tableATableB", cascade=CascadeType.All, orphanRemoval=true)
  private Set<TableB> tableBs = new LinkedHashSet<TableB>();

  public void addTableB(TableB b) {
    this.tableBs.add(b);
    b.setTableATableB(this);
  }
}

@Entity
@Table(name="TableB")
public class TableB {
  @Id
  // Generates via a sequence object
  private Integer id;

  @ManyToOne
  @JoinColumn(name="TABLE_J_ID")
  private TableJ tableJTableB;

  @ManyToOne
  @JoinColumn(name="TABLE_A_ID")
  private TableA tableATableB;

  @OneToMany(mapped="tableBTableC", cascade=CascadeType.All, orphanRemoval=true)
  private Set<TableC> tableCs = new LinkedHashSet<TableC>();

  public void addTableC(TableC c) {
    this.tableCs.add(c);
    b.setTableBTableC(this);
  }
}

@Entity
@Table(name="TableJ")
public class TableJ {
  @Id
  // Generates via a sequence object
  private Integer id;

  @OneToMany(mapped="tableJTableB", cascade=CascadeType.All)
  private Set<TableB> tableBs = new LinkedHashSet<TableB>();

  @OneToMany(mapped="tableJTableC", cascade=CascadeType.All)
  private Set<TableC> tableCs = new LinkedHashSet<TableC>();

  public void addTableB(TableB b) {
    this.tableBs.add(b);
    b.setTableJTableB(this);
  }

  public void addTableC(TableC c) {
    this.tableCs.add(c);
    b.setTableJTableC(this);
  }

}

@Entity
@Table(name="TableC")
public class TableC {
  @Id
  // Generates via a sequence object
  private Integer id;

  @ManyToOne
  @JoinColumn(name="TABLE_J_ID")
  private TableJ tableJTableC;

  @ManyToOne
  @JoinColumn(name="TABLE_B_ID")
  private TableB tableBTableC;

}    

Способ построения графа / дерева объектов:

TableA tableA = new TableA();

// Get a list of TableB's (bList) from an external call
// THIS BASICALLY ADDS TABLE B's to TABLE A and TABLE J
for (TableB b : bList) {
  TableJ j = new TableJ();
  j.addTableB(b);

  tableA.addTableB(b);

}

// Get a list of TableC's (cList) from an external call
// THIS BASICALLY ADDS TABLE C's to TABLE B and TABLE J
for (TableC c : cList) {

  for (TableB b : tableA.getTableBs()) {
    // Iterate through bList and find a match per some logic
    if (/* some logic to find a corresponding TableB */) {
      TableJ j = new TableJ();
      j.addTableC(c);

      b.addTableC(c);
    }

  }
}

Когда я выполняю код, я получаю следующую ошибку:

**Unexpected error occurred in scheduled task.: org.springframework.dao.InvalidDataAccessApiUsageException: org.hibernate.TransientPropertyValueException: object TableB.tableJTableB -> class TableJ; nested exception in java.lang.IllegalStateException: org.hibernate.TransientPropertyValueException: object references an unsaved transient instance - save the transient instance before flushing: TableB.tableJTableB -> class TableJ

Caused by: java.lang.IllegalStateException: org.hibernate.TransientPropertyValueException: object references an unsaved transient instance - save the transient instance before flushing: TableB.tableJTableB -> class TableJ**

Может кто-нибудь сказать мне, в чем проблема?

1 Ответ

0 голосов
/ 25 февраля 2019

Проблема:

Первый А вставлен.Тогда A_ID используется как FK в B. Вставлен.B также требуется J_ID FK, но J еще не был сохранен = переходный процесс.

Порядок, в котором JPA создает вставки, должен быть изменен.J_ID необходим для создания B, C.Сохраняйте J перед добавлением его в B, C.

for (TableB b : bList) {
  TableJ j = new TableJ();
  em.persist(j)
  j.addTableB(b);
  tableA.addTableB(b);
}

и другой цикл

...