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 选择失败的主要内容,如果未能解决你的问题,请参考以下文章

多对多关系中没有项目的实体的 JPQL

多对多查询 jpql

具有多对多关系的 JPQL 查询

返回涉及多对多关系的实体的 JPQL 查询

JPQL 基于条件的多对多连接

多对多 JPQL @Query 与 List 作为 Spring 中的参数