定义查询方法
Repository 代理有两种方法可以从方法名称派生出针对特定存储的查询:
-
直接从方法名称派生查询。
-
使用手动定义的查询。
可用选项取决于实际的存储。但是,必须有一个策略来决定创建什么实际的查询。下一节描述了可用的选项。
查询查找策略
以下策略可用于 Repository 基础设施解析查询。通过 XML 配置,你可以通过 query-lookup-strategy
属性在命名空间上配置策略。对于 Java 配置,你可以使用 EnableJpaRepositories
注解的 queryLookupStrategy
属性。某些策略可能不支持特定的数据存储。
-
CREATE
尝试从查询方法名称构造针对特定存储的查询。一般方法是移除方法名称中一组已知的公共前缀,并解析方法的其余部分。你可以在“查询创建”中阅读更多关于查询构造的内容。 -
USE_DECLARED_QUERY
尝试查找已声明的查询,如果找不到则抛出异常。查询可以通过注解或其他方式在某处定义。请查阅特定存储的文档以了解该存储的可用选项。如果在启动时 Repository 基础设施找不到该方法的已声明查询,则会失败。 -
CREATE_IF_NOT_FOUND
(默认)结合了CREATE
和USE_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);
}
解析查询方法名称分为主体 (subject) 和谓词 (predicate)。第一部分(如 find…By
, exists…By
)定义查询的主体,第二部分构成谓词。引入子句(主体)可以包含进一步的表达式。find
(或其他引入关键字)和 By
之间的任何文本都被认为是描述性的,除非使用了限制结果的关键字,例如 Distinct
用于为要创建的查询设置 distinct 标志,或者Top
/First
用于限制查询结果。
附录包含了完整的查询方法主体关键字列表以及查询方法谓词关键字,包括排序和字母大小写修饰符。然而,第一个 By
充当分隔符,指示实际条件谓词的开始。在非常基本的层面上,你可以定义实体属性的条件,并使用 And
和 Or
将它们连接起来。
解析方法的实际结果取决于你创建查询所针对的持久化存储。但是,有一些通用的注意事项:
-
表达式通常是属性遍历与可连接的运算符相结合。你可以使用
AND
和OR
组合属性表达式。属性表达式还支持诸如Between
,LessThan
,GreaterThan
和Like
等运算符。支持的运算符可能因数据存储而异,因此请查阅你的参考文档中相应的部分。 -
方法解析器支持为单个属性设置
IgnoreCase
标志(例如findByLastnameIgnoreCase(…)
),或为支持忽略大小写的所有类型属性设置此标志(通常是String
实例——例如findByLastnameAndFirstnameAllIgnoreCase(…)
)。是否支持忽略大小写可能因存储而异,因此请查阅参考文档中关于特定存储查询方法的有关章节。 -
你可以通过向引用属性的查询方法附加
OrderBy
子句并提供排序方向(Asc
或Desc
)来应用静态排序。要创建支持动态排序的查询方法,请参阅“分页、迭代大结果集、排序与限制”。
保留的方法名称
虽然派生的 Repository 方法按名称绑定到属性,但对于从基 Repository 继承的某些方法名称(针对 标识符 属性)来说,存在一些例外。这些 保留方法,如 CrudRepository#findById
(或仅 findById
),无论声明的方法中实际使用了什么属性名称,都指向 标识符 属性。
考虑以下域类型,它包含一个通过 @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 | 它通过使用 find 和 by 之间的描述性标记来指向 id 属性,以避免与 保留方法 冲突。 |
这种特殊行为不仅适用于查找方法,也适用于 exists
和 delete
方法。请参阅“Repository 查询关键字”以获取方法列表。
属性表达式
属性表达式只能引用托管实体的直接属性,如前面的示例所示。在查询创建时,你已经确保解析的属性是托管域类的属性。但是,你也可以通过遍历嵌套属性来定义约束。考虑以下方法签名:
List<Person> findByAddressZipCode(ZipCode zipCode);
假设一个 Person
拥有一个带有 ZipCode
的 Address
。在这种情况下,该方法会创建 x.address.zipCode
属性遍历。解析算法首先将整个部分(AddressZipCode
)解释为属性,并在域类中检查是否存在该名称(首字母小写)的属性。如果算法成功,则使用该属性。如果失败,算法会从右侧的驼峰命名部分将源拆分为头部和尾部,并尝试查找相应的属性 — 在我们的示例中是 AddressZip
和 Code
。如果算法找到了带有该头部的属性,它会取尾部并从那里继续向下构建树,以刚描述的方式拆分尾部。如果第一次拆分不匹配,算法会将拆分点向左移动(Address
,ZipCode
)并继续。
尽管这在大多数情况下应该可行,但算法有可能选择错误的属性。假设 Person
类也包含一个 addressZip
属性。算法会在第一轮拆分中就匹配到,选择错误的属性,并失败(因为 addressZip
的类型可能没有 code
属性)。
为了解决这种歧义,你可以在方法名称中使用 _
手动定义遍历点。因此,我们的方法名称将如下所示:
List<Person> findByAddress_ZipCode(ZipCode zipCode);
因为我们将下划线( |
以下划线开头的字段名称
字段名称可以以下划线开头,例如 全大写字段名称
全大写的字段名称可以照常使用。如果适用,嵌套路径需要通过 第二个字母大写的字段名称
字段名称由一个起始小写字母后跟一个大写字母组成,例如 路径歧义
在以下示例中,属性
由于首先考虑对属性进行直接匹配,任何潜在的嵌套路径都不会被考虑,算法会选择 |
返回集合或可迭代对象的 Repository 方法
返回多个结果的查询方法可以使用标准的 Java Iterable
, List
和 Set
。此外,我们还支持返回 Spring Data 的 Streamable
(Iterable
的自定义扩展)以及 Vavr 提供的集合类型。请参考附录中解释所有可能的查询方法返回类型的部分。
使用 Streamable 作为查询方法的返回类型
你可以使用 Streamable
作为 Iterable
或任何集合类型的替代品。它提供了方便的方法来访问非并行 Stream
(这是 Iterable
所缺少的),并且可以直接对元素进行 ….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。通常,这些类型是通过调用返回集合类型的方法并手动创建包装类型的实例来使用的。你可以避免这个额外的步骤,因为 Spring Data 允许你将这些包装类型用作查询方法的返回类型,前提是它们满足以下条件:
-
类型实现了
Streamable
接口。 -
类型提供了一个构造函数或名为
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 | 一个暴露了访问产品价格 API 的 Product 实体。 |
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 源类型 |
---|---|---|
|
|
|
|
|
|
|
|
|
你可以将第一列中的类型(或其子类型)用作查询方法的返回类型,并根据实际查询结果的 Java 类型(第三列)获得第二列中用作实现类型的类型。或者,你可以声明 Traversable
(Vavr 的 Iterable
等效项),然后我们将从实际返回值中派生出实现类。也就是说,一个 java.util.List
会转换为 Vavr 的 List
或 Seq
,一个 java.util.Set
会变成 Vavr 的 LinkedHashSet
Set
等等。
流式查询结果
你可以通过使用 Java 8 Stream<T>
作为返回类型来增量处理查询方法的结果。与将查询结果包装在 Stream
中不同,这里使用针对特定数据存储的方法来执行流式处理,如下例所示:
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 作为返回类型。 |
分页、迭代大结果集、排序与限制
要处理查询中的参数,请定义方法参数,如前例所示。此外,基础设施会识别某些特定类型,如 Pageable
, Sort
和 Limit
,以便动态地为你的查询应用分页、排序和限制。以下示例展示了这些功能:
Pageable
, Slice
, Sort
和 Limit
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 , Pageable 和 Limit 的 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
实例所需的额外元数据不会被创建(反过来,这意味着本来必要的额外计数查询不会被发出)。相反,它将查询限制为仅查找给定范围的实体。
要确定整个查询有多少页,你需要触发额外的计数查询。默认情况下,此查询是根据你实际触发的查询派生而来的。 |
特殊参数在查询方法中只能使用一次。
用于限制结果的关键字 |
哪种方法是合适的?
Spring Data 抽象提供的值或许最好地体现在下表中列出的可能的查询方法返回类型中。该表展示了你可以从查询方法返回的类型:
方法 | 获取的数据量 | 查询结构 | 约束 |
---|---|---|---|
所有结果。 |
单个查询。 |
查询结果可能会耗尽所有内存。获取所有数据可能非常耗时。 |
|
所有结果。 |
单个查询。 |
查询结果可能会耗尽所有内存。获取所有数据可能非常耗时。 |
|
分块处理(逐个或批量),取决于 |
通常使用游标的单个查询。 |
使用后必须关闭 Stream 以避免资源泄漏。 |
|
|
分块处理(逐个或批量),取决于 |
通常使用游标的单个查询。 |
存储模块必须提供响应式基础设施。 |
|
在 |
从 |
一个
|
|
在 |
从 |
通常需要成本高昂的
|
分页和排序
你可以通过使用属性名称来定义简单的排序表达式。你可以连接表达式将多个条件收集到一个表达式中。
Sort sort = Sort.by("firstname").ascending()
.and(Sort.by("lastname").descending());
为了更类型安全地定义排序表达式,请从要定义排序表达式的类型开始,并使用方法引用来定义要排序的属性。
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,你也可以使用生成的元模型类型来定义排序表达式:
QSort sort = QSort.by(QPerson.firstname.asc())
.and(QSort.by(QPerson.lastname.desc()));
限制查询结果
除了分页,还可以使用专用的 Limit
参数来限制结果大小。你还可以通过使用 First
或 Top
关键字来限制查询方法的结果,这两个关键字可以互换使用,但不能与 Limit
参数混用。你可以在 Top
或 First
后附加一个可选的数字值,以指定要返回的最大结果大小。如果省略该数字,则假定结果大小为 1。以下示例展示了如何限制查询大小:
Top
和 First
限制查询结果大小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 查询的数据存储的 Distinct
关键字。此外,对于将结果集限制为单个实例的查询,支持使用 Optional
关键字包装结果。
如果分页或分块应用于限制性查询分页(以及可用页数的计算),则它会在限制结果内部应用。
将结果限制与通过使用 Sort 参数实现的动态排序相结合,可以让你表达查询 'K' 个最小元素和 'K' 个最大元素的方法。 |