在插入/保存最后一个元素时,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/