JPQL 多对多 Dto 选择失败
Posted
技术标签:
【中文标题】JPQL 多对多 Dto 选择失败【英文标题】:JPQL Many-to-many Dto select failure 【发布时间】:2021-09-05 11:42:09 【问题描述】:我想编写 JPQL 查询以从我的 AppUserRepository 中选择 DoctorDto。 AppUser 和 Specialization 实体之间存在多对多的关系。这两个实体如下所示:
@Entity
@Table(name = "app_user")
public class AppUser
@ManyToMany
@JoinTable(name = "doctor_specialization", joinColumns = @JoinColumn(name = "app_user_id"), inverseJoinColumns = @JoinColumn(name = "specialization_id"))
private Set<Specialization> specializations;
@Entity
@Table(name="specialization")
@NoArgsConstructor
public class Specialization
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String name;
我创建了一个 DoctorDto 对象:
@Getter
@Setter
@NoArgsConstructor
public class DoctorDto
private Set<Specialization> specializations;
public DoctorDto(Set<Specialization> specializations)
this.specializations = specializations;
我想从我的 AppUserRepository 中选择所有具有专长的医生。为此,我编写了以下 JPQL 查询:
@Query("select new com.my-package.DoctorDto(u.specializations) from AppUser u")
List<DoctorDto> getDoctors();
但是,我收到一个错误:
com.microsoft.sqlserver.jdbc.SQLServerException: Incorrect syntax near the keyword 'as'.
Hibernate 查询如下所示:
Hibernate: select . as col_0_0_ from app_user appuser0_ inner join doctor_specialization specializa1_ on appuser0_.id=specializa1_.app_user_id inner join specialization specializa2_ on specializa1_.specialization_id=specializa2_.id
如您所见,Hibernate 似乎没有检测到特化。没有 Dto 映射,一切都可以正常工作。这个错误的原因是什么?我该如何解决?提前致谢。
【问题讨论】:
我认为您无法获得这样的收藏。您必须在查询中明确加入表 你能具体说明你将如何实现它吗?对于非 Dto 示例,例如:“从 AppUser u 中选择 u.specializations”,它可以正常工作。我尝试在查询中加入表,但没有成功。 试试这些变通方法。 ***.com/questions/52452550/… 【参考方案1】:这不可能像 bluelurker 已经回答的那样开箱即用,但我认为这是 Blaze-Persistence Entity Views 的完美用例。
我创建了该库以允许在 JPA 模型和自定义接口或抽象类定义模型之间轻松映射,例如 Spring Data Projections on steroids。这个想法是您按照自己喜欢的方式定义目标结构(域模型),并通过 JPQL 表达式将属性(getter)映射到实体模型。
使用 Blaze-Persistence Entity-Views 的用例的 DTO 模型可能如下所示:
@EntityView(AppUser.class)
public interface DoctorDto
@IdMapping
Long getId();
Set<SpecializationDto> getSpecializations();
@EntityView(Specialization.class)
interface SpecializationDto
@IdMapping
Long getId();
String getName();
查询是将实体视图应用于查询的问题,最简单的就是通过 id 进行查询。
DoctorDto a = entityViewManager.find(entityManager, DoctorDto.class, id);
Spring Data 集成让您可以像使用 Spring Data Projections 一样使用它:https://persistence.blazebit.com/documentation/entity-view/manual/en_US/index.html#spring-data-features
Page<DoctorDto> findAll(Pageable pageable);
最好的部分是,它只会获取实际需要的状态!
【讨论】:
以上是关于JPQL 多对多 Dto 选择失败的主要内容,如果未能解决你的问题,请参考以下文章