我的问题是Hibernate不能持久化实体中的嵌套实体。
考虑以下实体。
PollEntity
@Table(name = "\"poll\"")
@Data
@AllArgsConstructor
@NoArgsConstructor
@Entity
@EqualsAndHashCode(of = "id")
@Builder
public class PollEntity {
@Transient
public OptionEntity addOption(OptionEntity pollOptionEntity) {
if(options == null)
options = new HashSet<>();
options.add(pollOptionEntity);
pollOptionEntity.setPoll(this);
return pollOptionEntity;
}
@Transient
public OptionEntity dropOption(OptionEntity pollOptionEntity) {
options.remove(pollOptionEntity);
pollOptionEntity.setPoll(null);
return pollOptionEntity;
}
@Id
@Column(name = "id")
private UUID id;
@Column(name = "author")
@UUIDv4
private UUID author;
@Column(name = "poll_question")
@Size(max = 1000)
private String pollQuestion;
@OneToMany(fetch = FetchType.EAGER, mappedBy = "poll", cascade = CascadeType.MERGE)
@Builder.Default
@Valid
private Set<OptionEntity> options;
}
选项实体
@Table(name = "\"option\"")
@Data
@AllArgsConstructor
@NoArgsConstructor
@Entity
@EqualsAndHashCode(of = "id")
@Builder
public class OptionEntity {
@Id
@GeneratedValue(generator = "UUID")
@Column(name = "id")
@GenericGenerator(name = "UUID", strategy = "org.hibernate.id.UUIDGenerator")
private UUID id;
@JoinColumn(name = "poll_id")
@ManyToOne
private PollEntity poll;
@Column(name = "option")
@Size(max = 1000)
@NotNull
private String option;
}
而这里的服务方法。
@Transactional(rollbackFor = Throwable.class)
public void createPoll(@Valid PollEntity pollEntity) throws ValidationException {
validationService.validateOrThrow(pollEntity);
if (pollRepository.findById(pollEntity.getId()).isPresent())
throw new ValidationException("Invalid id", Map.of("id", "Poll with id (" + pollEntity.getId() + ") already exists"));
pollEntity = validationService.validateAndSave(pollRepository, pollEntity);
以及相应的测试
@Test
public void createPollTest() throws ValidationException {
var uuid = UUID.randomUUID();
var pollOption1 = OptionEntity.builder()
.option("Test option 1")
.build();
var pollOption2 = OptionEntity.builder()
.option("Test option 2")
.build();
var pollOption3 = OptionEntity.builder()
.option("Test option 3")
.build();
var poll = PollEntity.builder()
.id(uuid)
.pollQuestion("Test question")
.author(UUID.randomUUID())
.build();
poll.addOption(pollOption1);
poll.addOption(pollOption2);
poll.addOption(pollOption3);
pollService.createPoll(poll);
}
在数据库中的输出如下
投票
2e565f50-7cd4-4fc9-98cd-49e0f0964487 feae5781-ff07-4a21-9292-c11c4f1a047d Test question
选择权
c786fe5d-632d-4e94-95ef-26ab2af633e7 fc712242-8e87-41d8-93f2-ff0931020a4a Test option 1
和其余选项最后都没有坚持下来。
我也曾在单独的方法中使用过创建选项的方法
@Transactional(propagation= Propagation.REQUIRES_NEW, rollbackFor = Throwable.class)
public Set<OptionEntity> createOptions(@Valid Set<OptionEntity> pollOptionsEntities) throws ValidationException {
for (var pollOption : pollOptionsEntities) {
validationService.validateAndSave(pollOptionsRepository, pollOption);
}
return pollOptionsEntities;
}
和选项实体被生产出来,但由于持久化投票实体的错误,不得不从内置实体方法切换到持久化。
数据库模式是这样的。
CREATE TABLE "poll"
(
"id" UUID PRIMARY KEY,
"author" UUID NOT NULL,
"poll_question" VARCHAR(1000) NOT NULL
)
CREATE TABLE "option"
(
"id" UUID PRIMARY KEY,
"poll_id" UUID NOT NULL REFERENCES "poll" ("id") ON DELETE CASCADE
"option" VARCHAR(1000) NOT NULL
)
有什么可行的方法可以尝试?
UPD 1
我想出了这个实体的附加功能,它可以使实体持久化,不管实际值如何,而且在输出中仍然只有一个选项。到底做错了什么?
@Override
public boolean equals(Object object) {
if ( object == this ) {
return false;
}
if ( object == null || object.getClass() != getClass() ) {
return false;
}
final OptionEntity other = OptionEntity.class.cast( object );
if ( getId() == null && other.getId() == null ) {
return false;
}
else {
return false;
}
}
@Override
public int hashCode() {
final HashCodeBuilder hcb = new HashCodeBuilder( 17, 37 );
if ( id == null ) {
while (getOptions().iterator().hasNext())
hcb.append( getOptions().iterator().next() );
}
else {
hcb.append( id );
}
hcb.append( options );
return hcb.toHashCode();
}
所以答案都相当琐碎的一路长。
为了克服这个问题,唯一要做的就是把容器改。Set<OptionEntity>
改成 List<OptionEntity>
.
希望这不会产生一些难以攻克的bug,但如果可以--请添加评论。
因为在我的案例中,唯一性不是严格的要求,所以最后用了这个。
PollEntity:
@Table(name = "\"poll\"")
@Data
@AllArgsConstructor
@NoArgsConstructor
@Entity
@Builder
public class PollEntity {
@Transient
public OptionEntity addOption(OptionEntity pollOptionEntity) {
options.add(pollOptionEntity);
pollOptionEntity.setPoll(this);
return pollOptionEntity;
}
@Transient
public OptionEntity dropOption(OptionEntity pollOptionEntity) {
options.remove(pollOptionEntity);
pollOptionEntity.setPoll(null);
return pollOptionEntity;
}
@Id
@Column(name = "id")
private UUID id;
@Column(name = "status")
@Enumerated(EnumType.STRING)
@NotNull
private Status status;
@Column(name = "author")
@UUIDv4
private UUID author;
@Column(name = "poll_question")
@Size(max = 1000)
private String pollQuestion;
@OneToMany(fetch = FetchType.EAGER, mappedBy = "poll", cascade = CascadeType.MERGE)
@Builder.Default
@Valid
private List<OptionEntity> options = new ArrayList<>();
}