我在Spring Boot中设置了两个数据源。在控制器中,我想在两个控制器上都保留几个父子实体。它们几乎是完全相同的双向关系,但是第一个起作用,而第二个发出一个提交消息,但实际上并没有持久化子实体。 工作数据源中的实体
@Entity
public class Brand {
private Integer id;
private Set<Line> lines = new HashSet<Line>();
@Id
@GeneratedValue(strategy=GenerationType.AUTO)
@JsonIgnore
public Integer getId() {
return id;
}
@JsonIgnore
@OneToMany(mappedBy = "brand", cascade = CascadeType.ALL, orphanRemoval = true)
public Set<Line> getLines() {
return lines;
}
@Entity
public class Line {
private Integer id;
private Brand brand;
@Id
@GeneratedValue(strategy=GenerationType.AUTO)
public Integer getId() {
return id;
}
@ManyToOne
@JoinColumn(name = "brand")
public Brand getBrand() {
return brand;
}
虽然这些来自第二个数据源,但它保留了父级而不是子级
@Entity
public class User {
private String id;
private List<UserCommit> userCommitList = new ArrayList<UserCommit>();
@Id
@GeneratedValue(strategy=GenerationType.AUTO)
public Integer getId() {
return id;
}
}
@OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true)
public List<UserCommit> getUserCommitList() {
return userCommitList;
}
@Entity
public class UserCommit {
private Integer id;
private User user;
@Id
@GeneratedValue(strategy=GenerationType.AUTO)
public Integer getId() {
return id;
}
@ManyToOne
@JoinColumn(name = "user")
public User getUser() {
return user;
}
这是控制器有趣的部分
Brand brand = brandService.createFrom(fsDto.getBrand());
Line line = lineService.createFrom(fsDto.getLine(), brand);
AccessToken accToken = token.getAccount().getKeycloakSecurityContext().getToken();
try {
prodService.updateFrom(id, line, fsDto);
User user = userService.createFrom(accToken);
ucService.createFrom(user, id, "EDIT", "FIRST");
这些是服务方法
public Brand createFrom(String name) {
Brand foundBrand = findByName(name);
Brand brand = new Brand();
if (foundBrand == null) {
brand = brandRepo.save(brand);
brand.setName(name);
} else
brand = foundBrand;
return brand;
public Line createFrom(String name, Brand brand) {
Line foundLine = findByNameAndBrand(name, brand);
Line line = new Line();
if (foundLine == null) {
line.setName(name);
line.setBrand(brand);
line.getBrand().addLine(line);
} else
line = foundLine;
return line;
}
public User createFrom(AccessToken accToken) {
User user = findByKcId(accToken.getSubject());
if(user == null) {
user = new User();
user.setKcId(accToken.getSubject());
user.setName(accToken.getPreferredUsername());
userRepo.save(user);
}
return user;
}
public UserCommit createFrom(User user, Integer prodId, String ucType, String stage) {
UserCommit uc = new UserCommit();
uc.setUser(user);
uc.getUser().addUserCommit(uc);
uc.setProdId(prodId);
uc.setUCType(UserCommit.UCType.valueOf(ucType));
uc.setStage(UserCommit.Stage.valueOf(stage));
uc.setTime(LocalDateTime.now());
return uc;
}
编辑 我更新了代码,使两对交易变得更加相似,现在,对于第二个交易,我在惰性初始化中遇到了这个经典错误。
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: petmenu.entities.users.User.userCommitList, could not initialize proxy - no Session
真正使我发疯的是,品牌→行有效,而用户→用户提交无效。即使是跟踪JPA,我也无法理解为什么UserCommit提交时会话中不包含用户实体。
2020-05-26 16:51:54.258 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Found thread-bound EntityManager [SessionImpl(1767213597<open>)] for JPA transaction
2020-05-26 16:51:54.258 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Creating new transaction with name [petmenu.services.users.UserService.createFrom]: PROPAGATION_REQUIRED,ISOLATION_DEFAULT
2020-05-26 16:51:54.259 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Exposing JPA transaction as JDBC [org.springframework.orm.jpa.vendor.HibernateJpaDialect$HibernateConnectionHandle@41ca5759]
2020-05-26 16:52:01.687 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.EntityManagerFactoryUtils : Opening JPA EntityManager
2020-05-26 16:52:01.708 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] org.hibernate.SQL : select user0_.id as id1_0_, user0_.kc_id as kc_id2_0_, user0_.name as name3_0_ from user user0_ where user0_.kc_id=?
2020-05-26 16:52:01.708 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.h.type.descriptor.sql.BasicBinder : binding parameter [1] as [VARCHAR] - [80a3b4b1-00d1-4062-a7e5-1927b938c203]
2020-05-26 16:52:01.709 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.h.type.descriptor.sql.BasicExtractor : extracted value ([id1_0_] : [INTEGER]) - [2005]
2020-05-26 16:52:01.709 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.h.type.descriptor.sql.BasicExtractor : extracted value ([kc_id2_0_] : [VARCHAR]) - [80a3b4b1-00d1-4062-a7e5-1927b938c203]
2020-05-26 16:52:01.709 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.h.type.descriptor.sql.BasicExtractor : extracted value ([name3_0_] : [VARCHAR]) - [user1]
2020-05-26 16:52:01.709 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Triggering beforeCommit synchronization
2020-05-26 16:52:01.709 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Triggering beforeCompletion synchronization
2020-05-26 16:52:01.709 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Initiating transaction commit
2020-05-26 16:52:01.709 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Committing JPA transaction on EntityManager [SessionImpl(1767213597<open>)]
2020-05-26 16:53:06.630 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Triggering afterCommit synchronization
2020-05-26 16:53:06.671 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Triggering afterCompletion synchronization
2020-05-26 16:53:06.671 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Not closing pre-bound JPA EntityManager after transaction
2020-05-26 16:53:06.671 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Found thread-bound EntityManager [SessionImpl(1767213597<open>)] for JPA transaction
2020-05-26 16:53:06.671 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Creating new transaction with name [petmenu.services.users.UserCommitService.createFrom]: PROPAGATION_REQUIRED,ISOLATION_DEFAULT
2020-05-26 16:53:06.671 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Exposing JPA transaction as JDBC [org.springframework.orm.jpa.vendor.HibernateJpaDialect$HibernateConnectionHandle@323b85aa]
2020-05-26 16:53:17.438 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Triggering beforeCompletion synchronization
2020-05-26 16:53:17.472 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Initiating transaction rollback
2020-05-26 16:53:17.472 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Rolling back JPA transaction on EntityManager [SessionImpl(1767213597<open>)]
2020-05-26 16:53:17.472 TRACE 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Triggering afterCompletion synchronization
2020-05-26 16:53:17.472 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.s.orm.jpa.JpaTransactionManager : Not closing pre-bound JPA EntityManager after transaction
2020-05-26 16:53:17.473 DEBUG 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.j.s.OpenEntityManagerInViewInterceptor : Closing JPA EntityManager in OpenEntityManagerInViewInterceptor
2020-05-26 16:53:17.496 ERROR 78269 --- [http-nio-192.168.1.10-8080-exec-2] o.a.c.c.C.[.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: petmenu.entities.users.User.userCommitList, could not initialize proxy - no Session] with root cause
答案 0 :(得分:0)
我研究了很多有关事务管理的内容,(希望)找到了罪魁祸首。 但值得感谢的是每一项更正。
LSS
Spring Boot似乎仅为主要数据源注册了OpenEntityManagerInViewInterceptor
(我在网上找到了有关此信息,但在官方文档中却找不到任何信息)。
TL:DR
在运行LineService.createFrom
时,将会话绑定到整个视图并加载热切管理的实体的集合已被保留,然后新添加的Line在事务结束时被刷新。
在没有视图会话的对象上,我在UserCommitService.createFrom
事务中没有托管集合,然后没有显式调用UserCommitRepo.save
,UserCommit将保持瞬态状态,并且在事务提交时将被从刷新中静默丢弃。 / p>
我不知道我是否正确理解了所有内容,如果是这样,我希望有人在{{1}上解释CascadeType.PERSIST
(包含在CascadeType.ALL
中)的意义。 }}双向
关系。