java - Hibernate 用 NULL 覆盖列值?

标签 java mysql hibernate orm spring-data-jpa

在插入/保存最后一个元素时,hibernate 会用 NULL 覆盖前面元素的 ManyToOne 列。下面是详细的说明。

我正在创建一些“状态”对象并将它们保存到数据库中 -

        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]

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

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]

生成的数据库表如下 - 您可以看到 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 

和 Country 实体 -

@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);

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

关于java - Hibernate 用 NULL 覆盖列值?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57848954/

相关文章:

java - 使用动态添加的元素进行数据绑定(bind)

php - 数据列表在下拉列表中的空格后未显示完整名称

java - 无法打开 JDBC 连接以执行 DDL

java - Hibernate - EntityManager 管理

java - 如何使用 Selenium 默认启用地理位置?

java - 字符串类内部 - 如果使用 UTF-8,缓存字符偏移到字节关系

php - 只给登录用户删除选项

android - SQLite 外键从另一个表获取值

java - 如何使用HQL获取组合列名称

java - 如何在同一 Java EE 应用程序中同时配置 BASIC 和 FORM 身份验证方法