gpt4 book ai didi

java - Hibernate 用 NULL 覆盖列值?

转载 作者:行者123 更新时间:2023-11-29 05:01:39 26 4
gpt4 key购买 nike

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

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com