在Inserting/Saving的最后一个元素上,hibernate用NULL覆盖了早期元素的ManyToOne Column。以下是详细说明。

我正在创建几个"State"对象并将它们持久保存到DB中 -

        Country usa = new Country ("USA", "330000000");
        State mi = new State("MI", "11000000", usa);
        State ca = new State("CA", "39900000", usa);
        State fl = new State("FL", "21300000", usa);

        countryRepository.save(usa); 
        stateRepository.save(mi); 
        stateRepository.save(ca); 
        stateRepository.save(fl);

Hibernate按预期将"State"对象持久保存到DB中,直到它到达持久化最后一个"State"对象。这是它触发以下查询并将空值插入国家/地区字段的位置。

2019-09-09 11:45:37.390 DEBUG 9600 --- [  restartedMain] org.hibernate.SQL                        : 
    update
        state 
    set
        country_name=null 
    where
        country_name=?
2019-09-09 11:45:37.390 TRACE 9600 --- [  restartedMain] o.h.type.descriptor.sql.BasicBinder      : binding parameter [1] as [VARCHAR] - [USA]

所以在数据库方面,它导致所有先前持久的状态(表中的行)的country字段被设置为空。
然后它坚持最后的状态 -

2019-09-09 11:45:39.649 DEBUG 9600 --- [  restartedMain] org.hibernate.SQL                        : 
    insert 
    into
        state
        (country_name, population, name) 
    values
        (?, ?, ?)
2019-09-09 11:45:39.650 TRACE 9600 --- [  restartedMain] o.h.type.descriptor.sql.BasicBinder      : binding parameter [1] as [VARCHAR] - [USA]
2019-09-09 11:45:39.650 TRACE 9600 --- [  restartedMain] o.h.type.descriptor.sql.BasicBinder      : binding parameter [2] as [VARCHAR] - [21300000]
2019-09-09 11:45:39.650 TRACE 9600 --- [  restartedMain] o.h.type.descriptor.sql.BasicBinder      : binding parameter [3] as [VARCHAR] - [FL]

生成的DB表如下所示 - 您可以看到CA和MI的country_name的空值,但不能看到最后保存状态的FL。

mysql> select * from state;
+------+------------+--------------+
| name | population | country_name |
+------+------------+--------------+
| CA   | 39900000   | NULL         |
| FL   | 21300000   | USA          |
| MI   | 11000000   | NULL         |
+------+------------+--------------+

java实体对象如下:
国家实体 -

@Table(name="state")
@Entity
public class State {

    @Id
    @Column(name="name")
    String name;
    @Column(name="population") 
    String population;  

    @OneToMany(cascade=CascadeType.ALL, fetch=FetchType.EAGER)
    @JoinColumn(name="state_name")
    List<City> cities;

    @ManyToOne (cascade=CascadeType.ALL, fetch=FetchType.LAZY) 
    Country country;  
    //getters and setters 

和国家实体 -

@Table(name="country")
@Entity
public class Country {

    @Id
    @Column(name="name")
    String name;

    @Column(name="population")
    String population;

    @OneToMany(cascade=CascadeType.ALL, fetch=FetchType.EAGER)
    @JoinColumn(name="country_name")
    List<State> states; 
    //getters and setters

任何暗示为什么这种覆盖国家字段的行为除了最后一个状态之外的每个状态都具有空值。我不期望其他州的国家空值。

分析解答

如果您有CascadeType.ALL,则无需执行所有操作

countryRepository.save(usa); 
stateRepository.save(mi); 
stateRepository.save(ca); 
stateRepository.save(fl);

我建议您确保在州的构造函数中将状态添加到您所在国家/地区的list状态,然后再保留该国家/地区。此外,我不会使用FetchType.EAGER而是在实际需要时使用它,因为它可能会导致大量的SQL语句