定义查询方法

repository 代理有两种方法可以从方法名派生出特定于存储的查询

  • 直接从方法名派生查询。

  • 使用手动定义的查询。

可用的选项取决于实际的存储。但是,必须有一个策略来决定创建什么样的实际查询。下一节描述了可用的选项。

查询查找策略

repository 基础结构可以使用以下策略来解析查询。通过 XML 配置,你可以在命名空间中通过 query-lookup-strategy 属性配置策略。对于 Java 配置,你可以使用 EnableJpaRepositories 注解的 queryLookupStrategy 属性。某些策略可能不支持特定的数据存储。

  • CREATE 尝试从查询方法名构建特定于存储的查询。通常的做法是移除方法名中一组已知的特定前缀,然后解析方法的其余部分。你可以在“查询创建”中阅读更多关于查询构建的内容。

  • USE_DECLARED_QUERY 尝试查找已声明的查询,如果找不到则抛出异常。查询可以通过注解或其他方式定义。请参阅特定存储的文档以找到该存储可用的选项。如果在启动时 repository 基础结构没有找到针对该方法的已声明查询,则会失败。

  • CREATE_IF_NOT_FOUND(默认)结合了 CREATEUSE_DECLARED_QUERY。它首先查找已声明的查询,如果没有找到已声明的查询,它会创建一个基于方法名的自定义查询。这是默认的查找策略,因此,如果你没有显式配置任何内容,就会使用此策略。它允许通过方法名快速定义查询,同时也允许通过在需要时引入已声明的查询来对这些查询进行自定义调整。

查询创建

内置在 Spring Data repository 基础结构中的查询构建器机制对于构建针对 repository 实体进行约束的查询非常有用。

以下示例展示了如何创建多个查询

从方法名创建查询
interface PersonRepository extends Repository<Person, Long> {

  List<Person> findByEmailAddressAndLastname(EmailAddress emailAddress, String lastname);

  // Enables the distinct flag for the query
  List<Person> findDistinctPeopleByLastnameOrFirstname(String lastname, String firstname);
  List<Person> findPeopleDistinctByLastnameOrFirstname(String lastname, String firstname);

  // Enabling ignoring case for an individual property
  List<Person> findByLastnameIgnoreCase(String lastname);
  // Enabling ignoring case for all suitable properties
  List<Person> findByLastnameAndFirstnameAllIgnoreCase(String lastname, String firstname);

  // Enabling static ORDER BY for a query
  List<Person> findByLastnameOrderByFirstnameAsc(String lastname);
  List<Person> findByLastnameOrderByFirstnameDesc(String lastname);
}

解析查询方法名分为主题和谓词两部分。第一部分(find…By, exists…By)定义了查询的主题,第二部分构成了谓词。引入子句(主题)可以包含进一步的表达式。find(或其他引入关键字)和 By 之间的任何文本都被认为是描述性的,除非使用限制结果的关键字,例如 Distinct 来设置查询的去重标志,或者 Top/First 来限制查询结果

附录包含了查询方法主题关键字的完整列表以及查询方法谓词关键字,包括排序和大小写修饰符。然而,第一个 By 作为分隔符表示实际条件谓词的开始。在非常基本的层面,你可以定义实体属性上的条件,并使用 AndOr 将它们连接起来。

解析方法的实际结果取决于你创建查询所使用的持久化存储。然而,有一些需要注意的通用事项

  • 表达式通常是属性遍历与可以连接起来的运算符的组合。你可以使用 ANDOR 组合属性表达式。对于属性表达式,你还支持诸如 Between, LessThan, GreaterThanLike 等运算符。支持的运算符可能因数据存储而异,请查阅你的参考文档中相应的部分。

  • 方法解析器支持为单个属性设置 IgnoreCase 标志(例如,findByLastnameIgnoreCase(…)),或者为支持忽略大小写的类型(通常是 String 实例,例如 findByLastnameAndFirstnameAllIgnoreCase(…))的所有属性设置该标志。是否支持忽略大小写可能因存储而异,因此请查阅参考文档中特定于存储的查询方法的相关章节。

  • 你可以通过在查询方法后附加引用属性的 OrderBy 子句并提供排序方向(AscDesc)来应用静态排序。要创建支持动态排序的查询方法,请参阅“分页、迭代大结果集、排序和限制”。

保留方法名

虽然派生的 repository 方法按名称绑定到属性,但对于某些从基础 repository 继承的方法名,如果它们的目标是 identifier 属性,则存在一些例外。那些保留方法,例如 CrudRepository#findById(或仅 findById),无论在声明的方法中使用的实际属性名是什么,它们都指向 identifier 属性。

考虑以下域类型,它包含一个通过 @Id 标记为标识符的属性 pk 和一个名为 id 的属性。在这种情况下,你需要密切注意查找方法的命名,因为它们可能与预定义的方法签名冲突

class User {
  @Id Long pk;                          (1)

  Long id;                              (2)

  // …
}

interface UserRepository extends Repository<User, Long> {

  Optional<User> findById(Long id);     (3)

  Optional<User> findByPk(Long pk);     (4)

  Optional<User> findUserById(Long id); (5)
}
1 标识符属性(主键)。
2 一个名为 id 的属性,但不是标识符。
3 目标是 pk 属性(即标记有 @Id 并被认为是标识符的属性),因为它引用了 CrudRepository 的基础 repository 方法。因此,它不是一个使用 id 作为属性名推断出的派生查询,因为它属于保留方法之一。
4 按名称指向 pk 属性,因为它是一个派生查询。
5 通过使用 findby 之间的描述性标记来指向 id 属性,以避免与保留方法发生冲突。

这种特殊行为不仅适用于查找方法,也适用于 existsdelete 方法。请参阅“Repository 查询关键字”以获取方法列表。

属性表达式

属性表达式只能引用受管理的实体的直接属性,如前面示例所示。在创建查询时,你已经确保解析的属性是受管理的域类的一个属性。但是,你也可以通过遍历嵌套属性来定义约束。考虑以下方法签名

List<Person> findByAddressZipCode(ZipCode zipCode);

假设一个 Person 有一个包含 ZipCodeAddress。在这种情况下,该方法会创建 x.address.zipCode 属性遍历。解析算法首先将整个部分(AddressZipCode)解释为属性,并检查域类中是否存在以此名称(首字母未大写)命名的属性。如果算法成功,则使用该属性。如果失败,算法会从右侧的驼峰命名部分将源分割成头部和尾部,并尝试找到相应的属性——在我们的示例中是 AddressZipCode。如果算法找到了一个以该头部命名的属性,它将 എടു尾部并从那里继续构建树结构,按照刚才描述的方式分割尾部。如果第一次分割不匹配,算法会将分割点向左移动(Address, ZipCode)并继续。

尽管这在大多数情况下应该有效,但算法有可能选择错误的属性。假设 Person 类也有一个 addressZip 属性。算法会在第一次分割时就匹配上,选择错误的属性,然后失败(因为 addressZip 的类型可能没有 code 属性)。

为了解决这种歧义,你可以在方法名中使用 _ 来手动定义遍历点。所以我们的方法名如下

List<Person> findByAddress_ZipCode(ZipCode zipCode);

由于我们将下划线(_)视为保留字符,我们强烈建议遵循标准的 Java 命名约定(即,不在属性名中使用下划线,而是应用驼峰命名法)。

以下划线开头的字段名

字段名可能以下划线开头,例如 String _name。请确保保留 _,如同 _name 一样,并使用双下划线 __ 来分割嵌套路径,例如 user__name

大写字段名

全大写的字段名可以直接使用。如果适用,嵌套路径需要通过 _ 分割,例如 USER_name

第二个字母大写的字段名

字段名由一个开头小写字母后跟一个大写字母组成,例如 String qCode,可以通过以两个大写字母开头来解析,例如 QCode。请注意潜在的路径歧义。

路径歧义

在以下示例中,属性 qCodeq 的排列方式(其中 q 包含一个名为 code 的属性)为路径 QCode 造成了歧义。

record Container(String qCode, Code q) {}
record Code(String code) {}

由于首先考虑直接匹配属性,任何潜在的嵌套路径都不会被考虑,算法会选择 qCode 字段。为了选择 q 中的 code 字段,需要使用下划线表示法 Q_Code

返回集合或可迭代对象的 Repository 方法

返回多个结果的查询方法可以使用标准的 Java Iterable, ListSet。除此之外,我们还支持返回 Spring Data 的 StreamableIterable 的自定义扩展)以及由 Vavr 提供的集合类型。请参阅附录,其中解释了所有可能的查询方法返回类型

使用 Streamable 作为查询方法返回类型

你可以使用 Streamable 作为 Iterable 或任何集合类型的替代。它提供了方便的方法来访问非并行 StreamIterable 中缺失的功能),并且可以直接对元素进行 ….filter(…)….map(…) 操作,并将 Streamable 与其他对象连接起来。

使用 Streamable 组合查询方法结果
interface PersonRepository extends Repository<Person, Long> {
  Streamable<Person> findByFirstnameContaining(String firstname);
  Streamable<Person> findByLastnameContaining(String lastname);
}

Streamable<Person> result = repository.findByFirstnameContaining("av")
  .and(repository.findByLastnameContaining("ea"));

返回自定义 Streamable 包装器类型

为集合提供专用的包装器类型是一种常用的模式,用于为返回多个元素的查询结果提供 API。通常,这些类型通过调用返回类似集合类型的 repository 方法并手动创建包装器类型的实例来使用。你可以避免这个额外的步骤,因为 Spring Data 允许你将这些包装器类型用作查询方法的返回类型,如果它们满足以下条件

  1. 该类型实现了 Streamable 接口。

  2. 该类型暴露了一个构造函数或一个名为 of(…)valueOf(…) 的静态工厂方法,它们接受 Streamable 作为参数。

以下列表展示了一个示例

class Product {                                         (1)
  MonetaryAmount getPrice() { … }
}

@RequiredArgsConstructor(staticName = "of")
class Products implements Streamable<Product> {         (2)

  private final Streamable<Product> streamable;

  public MonetaryAmount getTotal() {                    (3)
    return streamable.stream()
      .map(Product::getPrice)
      .reduce(Money.of(0), MonetaryAmount::add);
  }


  @Override
  public Iterator<Product> iterator() {                 (4)
    return streamable.iterator();
  }
}

interface ProductRepository implements Repository<Product, Long> {
  Products findAllByDescriptionContaining(String text); (5)
}
1 一个 Product 实体,它暴露了访问产品价格的 API。
2 一个用于 Streamable<Product> 的包装器类型,可以通过使用 Products.of(…) 构建(使用 Lombok 注解创建的工厂方法)。接受 Streamable<Product> 的标准构造函数也可以。
3 包装器类型暴露了一个额外的 API,用于在 Streamable<Product> 上计算新值。
4 实现 Streamable 接口并委托给实际结果。
5 包装器类型 Products 可以直接用作查询方法的返回类型。你无需返回 Streamable<Product> 并在 repository 客户端中手动包装它。

支持 Vavr 集合

Vavr 是一个在 Java 中拥抱函数式编程概念的库。它附带了一组自定义的集合类型,你可以将它们用作查询方法的返回类型,如下表所示

Vavr 集合类型 使用的 Vavr 实现类型 有效的 Java 源类型

io.vavr.collection.Seq

io.vavr.collection.List

java.util.Iterable

io.vavr.collection.Set

io.vavr.collection.LinkedHashSet

java.util.Iterable

io.vavr.collection.Map

io.vavr.collection.LinkedHashMap

java.util.Map

你可以使用第一列中的类型(或其子类型)作为查询方法的返回类型,并根据实际查询结果的 Java 类型(第三列)获得第二列中的类型作为实现类型。另外,你可以声明 Traversable(Vavr 的 Iterable 等价物),然后我们从实际返回值派生出实现类。也就是说,一个 java.util.List 会被转换为 Vavr 的 ListSeq,一个 java.util.Set 会变成 Vavr 的 LinkedHashSet Set,等等。

流式处理查询结果

你可以使用 Java 8 的 Stream<T> 作为返回类型,以增量方式处理查询方法的结果。与将查询结果包装到 Stream 中不同,这里使用特定于数据存储的方法来执行流式处理,如下例所示

使用 Java 8 的 Stream<T> 流式处理查询结果
@Query("select u from User u")
Stream<User> findAllByCustomQueryAndStream();

Stream<User> readAllByFirstnameNotNull();

@Query("select u from User u")
Stream<User> streamAllPaged(Pageable pageable);
Stream 可能包装了底层特定于数据存储的资源,因此在使用后必须关闭。你可以使用 close() 方法手动关闭 Stream,或者使用 Java 7 的 try-with-resources 块,如下例所示
try-with-resources 块中处理 Stream<T> 结果
try (Stream<User> stream = repository.findAllByCustomQueryAndStream()) {
  stream.forEach(…);
}
并非所有 Spring Data 模块目前都支持 Stream<T> 作为返回类型。

异步查询结果

你可以使用Spring 的异步方法运行能力来异步运行 repository 查询。这意味着方法在调用后立即返回,而实际的查询在提交到 Spring TaskExecutor 的任务中发生。异步查询与响应式查询不同,不应混用。有关响应式支持的更多详细信息,请参阅特定于存储的文档。以下示例展示了一些异步查询

@Async
Future<User> findByFirstname(String firstname);               (1)

@Async
CompletableFuture<User> findOneByFirstname(String firstname); (2)
1 使用 java.util.concurrent.Future 作为返回类型。
2 使用 Java 8 的 java.util.concurrent.CompletableFuture 作为返回类型。

分页、迭代大结果集、排序和限制

要在查询中处理参数,请像前面的示例中那样定义方法参数。除此之外,基础结构还识别某些特定类型,如 PageableSortLimit,以动态地将分页、排序和限制应用于你的查询。以下示例展示了这些功能

在查询方法中使用 Pageable, Slice, SortLimit
Page<User> findByLastname(String lastname, Pageable pageable);

Slice<User> findByLastname(String lastname, Pageable pageable);

List<User> findByLastname(String lastname, Sort sort);

List<User> findByLastname(String lastname, Sort sort, Limit limit);

List<User> findByLastname(String lastname, Pageable pageable);
接受 Sort, PageableLimit 的 API 期望传递非 null 值给方法。如果你不想应用任何排序或分页,请使用 Sort.unsorted(), Pageable.unpaged()Limit.unlimited()

第一个方法允许你将 org.springframework.data.domain.Pageable 实例传递给查询方法,以便动态地为静态定义的查询添加分页。一个 Page 知道元素的总数和可用的总页数。它通过基础结构触发一个计数查询来计算总数。由于这可能很昂贵(取决于使用的存储),你可以改为返回一个 Slice。一个 Slice 只知道是否有下一个 Slice 可用,这在遍历较大的结果集时可能就足够了。

排序选项也通过 Pageable 实例处理。如果你只需要排序,请向你的方法添加一个 org.springframework.data.domain.Sort 参数。正如你所见,返回一个 List 也是可能的。在这种情况下,构建实际 Page 实例所需的额外元数据不会被创建(反过来,这意味着不会发出本来必要的额外计数查询)。相反,它将查询限制为仅查找给定范围的实体。

要了解一个完整查询有多少页,你必须触发一个额外的计数查询。默认情况下,这个查询是从你实际触发的查询派生出来的。

特殊参数在一个查询方法中只能使用一次。
上面描述的一些特殊参数是互斥的。请考虑以下无效参数组合列表。

参数 示例 原因

PageableSort

findBy…​(Pageable page, Sort sort)

Pageable 已经定义了 Sort

PageableLimit

findBy…​(Pageable page, Limit limit)

Pageable 已经定义了限制。

用于限制结果的 Top 关键字可以与 Pageable 一起使用,其中 Top 定义了结果的总最大数量,而 Pageable 参数可以减少这个数量。

哪种方法是合适的?

Spring Data 抽象提供的值或许最好地体现在下表中列出的可能的查询方法返回类型。下表显示了你可以从查询方法返回哪些类型

表 1. 消费大型查询结果
方法 获取的数据量 查询结构 限制/约束

List<T>

所有结果。

单个查询。

查询结果可能会耗尽所有内存。获取所有数据可能非常耗时。

Streamable<T>

所有结果。

单个查询。

查询结果可能会耗尽所有内存。获取所有数据可能非常耗时。

Stream<T>

分块(逐个或批量)取决于 Stream 的消费方式。

通常使用游标的单个查询。

使用后必须关闭 Stream 以避免资源泄露。

Flux<T>

分块(逐个或批量),取决于 Flux 的消费方式。

通常使用游标的单个查询。

存储模块必须提供响应式基础结构。

Slice<T>

位于 Pageable.getOffset() 处的 Pageable.getPageSize() + 1

一个或多个查询,从 Pageable.getOffset() 开始获取数据并应用限制。

一个 Slice 只能导航到下一个 Slice

  • Slice 提供了是否有更多数据可供获取的详细信息。

  • 当偏移量太大时,基于偏移量的查询会变得低效,因为数据库仍然必须具体化完整的结果。

  • Window 提供了是否有更多数据可供获取的详细信息。

  • 当偏移量太大时,基于偏移量的查询会变得低效,因为数据库仍然必须具体化完整的结果。

Page<T>

位于 Pageable.getOffset() 处的 Pageable.getPageSize()

一个或多个查询,从 Pageable.getOffset() 开始并应用限制。此外,可能需要 COUNT(…) 查询来确定元素总数。

通常,需要执行成本高昂的 COUNT(…) 查询。

  • 当偏移量太大时,基于偏移量的查询会变得低效,因为数据库仍然必须具体化完整的结果。

分页和排序

你可以使用属性名定义简单的排序表达式。你可以连接表达式,将多个条件收集到一个表达式中。

定义排序表达式
Sort sort = Sort.by("firstname").ascending()
  .and(Sort.by("lastname").descending());

为了更类型安全地定义排序表达式,请首先指定要定义排序表达式的类型,然后使用方法引用来定义要排序的属性。

使用类型安全 API 定义排序表达式
TypedSort<Person> person = Sort.sort(Person.class);

Sort sort = person.by(Person::getFirstname).ascending()
  .and(person.by(Person::getLastname).descending());
TypedSort.by(…) 使用运行时代理(通常使用 CGlib),在使用 Graal VM Native 等工具进行原生镜像编译时可能会产生干扰。

如果你的存储实现支持 Querydsl,你也可以使用生成的元模型类型来定义排序表达式

使用 Querydsl API 定义排序表达式
QSort sort = QSort.by(QPerson.firstname.asc())
  .and(QSort.by(QPerson.lastname.desc()));

限制查询结果

除了分页之外,还可以使用专用的 Limit 参数限制结果集大小。你还可以使用 FirstTop 关键字限制查询方法的结果,这两个关键字可以互换使用,但不能与 Limit 参数混用。你可以在 TopFirst 后面附加一个可选的数值来指定要返回的最大结果集大小。如果省略该数值,则假定结果集大小为 1。以下示例展示了如何限制查询大小

使用 TopFirst 限制查询结果集大小
List<User> findByLastname(String lastname, Limit limit);

User findFirstByOrderByLastnameAsc();

User findTopByLastnameOrderByAgeDesc(String lastname);

Page<User> queryFirst10ByLastname(String lastname, Pageable pageable);

Slice<User> findTop3By(Pageable pageable);

List<User> findFirst10ByLastname(String lastname, Sort sort);

List<User> findTop10ByLastname(String lastname, Pageable pageable);

限制表达式也支持 Distinct 关键字,适用于支持去重查询的数据存储。此外,对于将结果集限制为一个实例的查询,支持使用 Optional 关键字包装结果。

如果对限制性查询应用分页或切片,分页(以及可用页数的计算)将在有限结果范围内应用。

结合使用 Sort 参数进行动态排序来限制结果,可以让你表达查询方法,获取“K”个最小元素以及“K”个最大元素。