Skip to content

Commit b1aeec1

Browse files
committed
Add documentation for the fluent API for Query By Example.
Closes #2654 Original pull request #2655 See #2228
1 parent 74e5261 commit b1aeec1

File tree

1 file changed

+32
-2
lines changed

1 file changed

+32
-2
lines changed

src/main/asciidoc/query-by-example.adoc

+32-2
Original file line numberDiff line numberDiff line change
@@ -13,13 +13,15 @@ In fact, Query by Example does not require you to write queries by using store-s
1313
[[query-by-example.usage]]
1414
== Usage
1515

16-
The Query by Example API consists of three parts:
16+
The Query by Example API consists of four parts:
1717

1818
* Probe: The actual example of a domain object with populated fields.
1919
* `ExampleMatcher`: The `ExampleMatcher` carries details on how to match particular fields.
2020
It can be reused across multiple Examples.
2121
* `Example`: An `Example` consists of the probe and the `ExampleMatcher`.
2222
It is used to create the query.
23+
* `FetchableFluentQuery`: A `FetchableFluentQuery` offers a fluent API, that allows further customization of a query derived from an `Example`.
24+
Using the fluent API allows you to specify ordering, projection and result processing for your query.
2325

2426
Query by Example is well suited for several use cases:
2527

@@ -56,7 +58,8 @@ The preceding example shows a simple domain object.
5658
You can use it to create an `Example`.
5759
By default, fields having `null` values are ignored, and strings are matched by using the store specific defaults.
5860

59-
NOTE: Inclusion of properties into a Query by Example criteria is based on nullability. Properties using primitive types (`int`, `double`, …) are always included unless <<query-by-example.matchers,ignoring the property path>>.
61+
NOTE: Inclusion of properties into a Query by Example criteria is based on nullability.
62+
Properties using primitive types (`int`, `double`, …) are always included unless <<query-by-example.matchers,ignoring the property path>>.
6063

6164
Examples can be built by either using the `of` factory method or by using <<query-by-example.matchers,`ExampleMatcher`>>. `Example` is immutable.
6265
The following listing shows a simple Example:
@@ -70,6 +73,7 @@ person.setFirstname("Dave"); <2>
7073
7174
Example<Person> example = Example.of(person); <3>
7275
----
76+
7377
<1> Create a new instance of the domain object.
7478
<2> Set the properties to query.
7579
<3> Create the `Example`.
@@ -115,6 +119,7 @@ ExampleMatcher matcher = ExampleMatcher.matching() <3>
115119
Example<Person> example = Example.of(person, matcher); <7>
116120
117121
----
122+
118123
<1> Create a new instance of the domain object.
119124
<2> Set properties.
120125
<3> Create an `ExampleMatcher` to expect all values to match.
@@ -186,3 +191,28 @@ The following table describes the scope of the various `ExampleMatcher` settings
186191
| Property path
187192

188193
|===
194+
195+
[[query-by-example.fluent]]
196+
== Fluent API
197+
198+
`QueryByExampleExecutor` offers one more method, which we did not mention so far: `<S extends T, R> R findBy(Example<S> example, Function<FluentQuery.FetchableFluentQuery<S>, R> queryFunction)`.
199+
As with other methods it executes a query derived from an `Example`.
200+
But with the second argument you can control aspects of that execution, that you can't control dynamically otherwise.
201+
You do so by invoking the various methods of the `FetchableFluentQuery` in the second argument.
202+
`sortBy` allows you to specify an ordering for your result.
203+
`as` allows you to specify the type to which you want the result to be transformed.
204+
`project` limits the attributes queried.
205+
`first`, `firstValue`, `one`, `oneValue`, `all`, `page`, `stream`, `count`, `exists` define what kind of result you'll get and also how the query behaves when more than the expected number of results are available.
206+
207+
208+
.Use the fluent API to get the last of potentially many results, ordered by lastname.
209+
====
210+
[source,java]
211+
----
212+
Optional<Person> match = repository.findBy(example,
213+
q -> q
214+
.sortBy(Sort.by("lastname").descending())
215+
.first()
216+
);
217+
----
218+
====

0 commit comments

Comments
 (0)