
在使用Spring Data JPA时,我们通常会通过@Query注解来定义自定义查询。这里需要明确区分两种主要的查询语言:
原始问题中出现的错误在于,尝试在nativeQuery=true的设置下,使用了类似JPQL的构造器表达式(new egecoskun121.com.crm.model.entity.Product(...))和SQL风格的子查询语法({SELECT PRODUCT_ID FROM USERS_PRODUCTS WHERE USER_ID=:id }),这种混淆导致了查询失败。当nativeQuery为true时,查询字符串必须是纯粹的SQL,不能包含JPQL特有的语法,反之亦然。
对于涉及实体间关联关系的查询,JPQL是首选且更优雅的解决方案。JPA通过实体类中的@OneToMany、@ManyToOne等注解自动管理这些关系。
考虑以下两个实体:User 和 Product,它们之间存在一对多的关系,即一个用户可以拥有多个产品。
Product 实体
import jakarta.persistence.*;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.hibernate.annotations.CreationTimestamp;
import org.hibernate.annotations.UpdateTimestamp;
import java.math.BigDecimal;
import java.sql.Timestamp;
import jakarta.validation.constraints.Size;
@Entity
@Data
@Table(name = "product")
@NoArgsConstructor
@AllArgsConstructor
public class Product {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@CreationTimestamp
@Column(updatable = false)
private Timestamp createdDate;
@UpdateTimestamp
private Timestamp lastModifiedDate;
private String imageURL;
private Long productCode;
@Size(min = 3,max = 100)
private String productName;
@Size(min = 5,max = 100)
private String details;
private BigDecimal price;
@Enumerated(EnumType.STRING) // 假设 ProductCategory 是枚举类型
private ProductCategory productCategory;
// ... 其他字段
}
// 假设 ProductCategory 是一个枚举
enum ProductCategory {
ELECTRONICS, BOOKS, CLOTHING
}User 实体
import jakarta.persistence.*;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.hibernate.annotations.CreationTimestamp;
import org.hibernate.annotations.UpdateTimestamp;
import java.sql.Timestamp;
import java.util.List;
import jakarta.validation.constraints.NotNull;
import jakarta.validation.constraints.Size;
@Data
@AllArgsConstructor
@NoArgsConstructor
@Entity
@Table(name = "users")
public class User {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@Column(unique = true,nullable = false)
private String phoneNumber;
@Size(min = 5, max = 25, message = "Username length should be between 5 and 25 characters")
@Column(unique = true, nullable = false)
private String userName;
@CreationTimestamp
@Column(updatable = false)
private Timestamp createdDate;
@UpdateTimestamp
private Timestamp lastModifiedDate;
@Column(unique = true, nullable = false)
@NotNull
private String email;
@Size(min = 5, message = "Minimum password length: 5 characters")
@NotNull
private String password;
// User 和 Product 之间的一对多关系
// 默认情况下,JPA会为 @OneToMany 关系创建一张中间表 (users_products)
// 如果没有mappedBy属性,则由拥有方(这里是User)管理关系
@OneToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL, orphanRemoval = true)
@JoinTable( // 明确指定中间表,或者让JPA自动生成
name = "users_products",
joinColumns = @JoinColumn(name = "user_id"),
inverseJoinColumns = @JoinColumn(name = "product_id")
)
private List<Product> products;
// ... 其他字段和关系
// @Transient 表示此字段不映射到数据库
@Transient
@OneToMany(fetch = FetchType.LAZY, mappedBy = "product", cascade = CascadeType.ALL, orphanRemoval = true)
private List<ProductInquiry> productInquiries; // 假设 ProductInquiry 实体存在
@Enumerated(EnumType.STRING) // 假设 Role 是枚举类型
private Role role;
}
// 假设 Role 是一个枚举
enum Role {
USER, ADMIN
}要查询特定用户下的所有产品,最简洁且推荐的方式是使用JPQL的JOIN操作符。
JPA Repository 接口
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;
import org.springframework.stereotype.Repository;
import java.util.List;
@Repository
public interface UserRepository extends JpaRepository<User, Long> {
/**
* 根据用户ID查询该用户关联的所有产品。
* 使用JPQL的JOIN操作,从User实体开始,通过其products集合关联到Product实体。
*
* @param id 用户的ID
* @return 属于该用户的所有产品列表
*/
@Query("SELECT p FROM User u JOIN u.products p WHERE u.id = :id")
List<Product> findAllProductsByUserId(@Param("id") Long id);
}JPQL 查询解析:SELECT p FROM User u JOIN u.products p WHERE u.id = :id
这种方式的优点是:
尽管JPQL是处理实体关联查询的首选,但在以下特定场景中,原生SQL查询可能更为合适:
当使用原生SQL并希望将其结果映射到Java对象时,有几种方法:
示例(原生SQL映射到DTO,仅作演示,不推荐用于本场景)
假设有一个DTO ProductDTO:
public class ProductDTO {
private Long id;
private String productName;
private BigDecimal price;
public ProductDTO(Long id, String productName, BigDecimal price) {
this.id = id;
this.productName = productName;
this.price = price;
}
// Getters and Setters
}你可以定义一个@SqlResultSetMapping:
@NamedNativeQuery(
name = "User.findProductsNative",
query = "SELECT p.ID, p.PRODUCT_NAME, p.PRICE FROM PRODUCT p JOIN USERS_PRODUCTS up ON p.ID = up.PRODUCT_ID WHERE up.USER_ID = :id",
resultSetMapping = "ProductDTOMapping"
)
@SqlResultSetMapping(
name = "ProductDTOMapping",
classes = @ConstructorResult(
targetClass = ProductDTO.class,
columns = {
@ColumnResult(name = "ID", type = Long.class),
@ColumnResult(name = "PRODUCT_NAME", type = String.class),
@ColumnResult(name = "PRICE", type = BigDecimal.class)
}
)
)
@Entity // @SqlResultSetMapping 通常与一个实体关联,或者定义在orm.xml中
// 实际应用中,如果映射到DTO,这个Mapping可以放在任何一个实体上,或者通过orm.xml配置
// 为了简化,这里假设它放在User实体上
public class User { /* ... */ }然后在Repository中使用:
public interface UserRepository extends JpaRepository<User, Long> {
@Query(nativeQuery = true) // 或者直接使用 @NamedNativeQuery 的名称
List<ProductDTO> findProductsNative(@Param("id") Long id);
}显然,相比于JPQL的简洁,原生SQL在实体映射方面要复杂得多。因此,除非有明确的理由,否则应优先选择JPQL。
通过遵循这些最佳实践,开发者可以编写出高效、健鲁且易于维护的JPA查询代码。
以上就是如何使用JPA Repository通过JPQL查询关联实体的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号