有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java JPA在多对多关系中按列排序

我的问题比第一眼看到的要复杂一些。让我来解释一下。 我有两个具有单向多人关系的实体:

public class Category implements Serializable {

    @Id
    @Column(name = "ct_id")
    @SequenceGenerator(name = "category_ct_id_seq", sequenceName = "category_ct_id_seq", allocationSize = 1)
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "category_ct_id_seq")
    private Long id;

    @Column(name = "ct_name_key")
    private String nameKey;

    @ManyToMany
    @JoinTable(name = "category_2_country",
            joinColumns = @JoinColumn(name = "c2c_category_id"),
            inverseJoinColumns = @JoinColumn(name = "c2c_country_id"))
    private Set<Country> countries;
}

public class Country {

    @Id
    @SequenceGenerator(name = "country_c_id_seq", sequenceName = "country_c_id_seq", allocationSize = 1)
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "country_c_id_seq")
    @Column(name = "c_id")
    private Long id;

    @Column(name = "c_name")
    private String name;

    @Column(name = "c_iso_2_alpha")
    private String isoAlpha2Code;

我通过国家的ISO代码检索类别集合(实际上是所有类别,与请求的国家有关)。为此,我编写了一份规范:

public class CategorySpecification implements Specification<Category> {

    Country matchCountry;

    @Override
    public Predicate toPredicate(Root<Category> root, CriteriaQuery<?> criteriaQuery, CriteriaBuilder builder) {

        return builder.isMember(matchCountry, root.get("countries"));

    }
}

这个很好用。问题是,我需要为每个国家的类别设定优先顺序。 可传递表如下所示: enter image description here 因此,当我检索给定国家的类别集合时,我希望它按照这个sort_order列进行排序

相应的SQL查询看起来是这样的:

SELECT *
FROM category
  JOIN category_2_country ON category.ct_id = category_2_country.c2c_category_id
WHERE c2c_country_id = ?
ORDER BY sort_order;

但我应该如何在我的JPA实体中实现这一点?我应该如何将其保存回数据库,如果这个顺序需要动态更改(例如,从管理面板)

编辑: 数据库中与实体对应的表:

国家队 Country

以及类别: enter image description here

编辑2: 我已经通过本机查询实现了预期的行为,如下所示:

@Repository
public interface CategoryRepository extends JpaRepository<Category, Long>, JpaSpecificationExecutor<Category> {

    @Query(value = "SELECT ct.*\n" +
            "FROM category ct\n" +
            "  JOIN category_2_country c2c ON ct.ct_id = c2c.c2c_category_id\n" +
            "  JOIN country c ON c2c.c2c_country_id = c.c_id\n" +
            "WHERE c.c_iso_2_alpha = :countryIso\n" +
            "ORDER BY c2c.sort_order", nativeQuery = true)
    List<Category> findCategoriesForCountryOrdered(@Param("countryIso") String countryIso);
}

但仍然希望有一个更JPA/Hibernate的方式来做这件事


共 (1) 个答案

  1. # 1 楼答案

    在使用规范时,您使用的是JpaRepositoryJpaSpecificationExecutorJpaSpecificationExecutorapi为我们提供了一种方法findAll(Specification<T> spec, Sort sort)

    使用Sort类,可以指定要按哪个属性(和方向)对结果列表进行排序

    以下是文档:Sort

    编辑: 经过一些搜索后——不幸的是,如果不将关联表映射到hibernate,就无法按sort_order属性排序。这与JPQL有关,它只能在实体中的映射属性上工作——如果不映射它们,它就无法访问数据库中的表

    因此,您可以映射您的关联表并创建规范,将这两个表连接起来,并按sort_order创建排序