查询查找策略

Elasticsearch 模块支持所有基本的查询构建功能,如字符串查询、本机搜索查询、基于条件的查询或从方法名称派生的查询。Spring中文文档

声明的查询

从方法名称派生查询并不总是足够的,并且/或可能导致方法名称不可读。 在这种情况下,可以使用注释(请参阅使用@Query注释)。@QuerySpring中文文档

查询创建

通常,Elasticsearch 的查询创建机制的工作方式如定义查询方法中所述。 下面是一个简短的示例,说明了 Elasticsearch 查询方法的转换方式:Spring中文文档

例 1.从方法名称创建查询
interface BookRepository extends Repository<Book, String> {
  List<Book> findByNameAndPrice(String name, Integer price);
}

上面的方法名称将被转换为以下 Elasticsearch json 查询Spring中文文档

{
    "query": {
        "bool" : {
            "must" : [
                { "query_string" : { "query" : "?", "fields" : [ "name" ] } },
                { "query_string" : { "query" : "?", "fields" : [ "price" ] } }
            ]
        }
    }
}

下面显示了 Elasticsearch 支持的关键字列表。Spring中文文档

表 1.方法名称中支持的关键字
关键词 样本 Elasticsearch 查询字符串

AndSpring中文文档

findByNameAndPriceSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } }, { "query_string" : { "query" : "?", "fields" : [ "price" ] } } ] } }}Spring中文文档

OrSpring中文文档

findByNameOrPriceSpring中文文档

{ "query" : { "bool" : { "should" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } }, { "query_string" : { "query" : "?", "fields" : [ "price" ] } } ] } }}Spring中文文档

IsSpring中文文档

findByNameSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } } ] } }}Spring中文文档

NotSpring中文文档

findByNameNotSpring中文文档

{ "query" : { "bool" : { "must_not" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } } ] } }}Spring中文文档

BetweenSpring中文文档

findByPriceBetweenSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : ?, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

LessThanSpring中文文档

findByPriceLessThanSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : null, "to" : ?, "include_lower" : true, "include_upper" : false } } } ] } }}Spring中文文档

LessThanEqualSpring中文文档

findByPriceLessThanEqualSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : null, "to" : ?, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

GreaterThanSpring中文文档

findByPriceGreaterThanSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : null, "include_lower" : false, "include_upper" : true } } } ] } }}Spring中文文档

GreaterThanEqualSpring中文文档

findByPriceGreaterThanEqualSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : null, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

BeforeSpring中文文档

findByPriceBeforeSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : null, "to" : ?, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

AfterSpring中文文档

findByPriceAfterSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : null, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

LikeSpring中文文档

findByNameLikeSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?*", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

StartingWithSpring中文文档

findByNameStartingWithSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?*", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

EndingWithSpring中文文档

findByNameEndingWithSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "*?", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

Contains/ContainingSpring中文文档

findByNameContainingSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "*?*", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

In(当批注为 FieldType.Keyword 时)Spring中文文档

findByNameIn(Collection<String>names)Spring中文文档

{ "query" : { "bool" : { "must" : [ {"bool" : {"must" : [ {"terms" : {"name" : ["?","?"]}} ] } } ] } }}Spring中文文档

InSpring中文文档

findByNameIn(Collection<String>names)Spring中文文档

{ "query": {"bool": {"must": [{"query_string":{"query": "\"?\" \"?\"", "fields": ["name"]}}]}}}Spring中文文档

NotIn(当批注为 FieldType.Keyword 时)Spring中文文档

findByNameNotIn(Collection<String>names)Spring中文文档

{ "query" : { "bool" : { "must" : [ {"bool" : {"must_not" : [ {"terms" : {"name" : ["?","?"]}} ] } } ] } }}Spring中文文档

NotInSpring中文文档

findByNameNotIn(Collection<String>names)Spring中文文档

{"query": {"bool": {"must": [{"query_string": {"query": "NOT(\"?\" \"?\")", "fields": ["name"]}}]}}}Spring中文文档

TrueSpring中文文档

findByAvailableTrueSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "true", "fields" : [ "available" ] } } ] } }}Spring中文文档

FalseSpring中文文档

findByAvailableFalseSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "false", "fields" : [ "available" ] } } ] } }}Spring中文文档

OrderBySpring中文文档

findByAvailableTrueOrderByNameDescSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "true", "fields" : [ "available" ] } } ] } }, "sort":[{"name":{"order":"desc"}}] }Spring中文文档

ExistsSpring中文文档

findByNameExistsSpring中文文档

{"query":{"bool":{"must":[{"exists":{"field":"name"}}]}}}Spring中文文档

IsNullSpring中文文档

findByNameIsNullSpring中文文档

{"query":{"bool":{"must_not":[{"exists":{"field":"name"}}]}}}Spring中文文档

IsNotNullSpring中文文档

findByNameIsNotNullSpring中文文档

{"query":{"bool":{"must":[{"exists":{"field":"name"}}]}}}Spring中文文档

IsEmptySpring中文文档

findByNameIsEmptySpring中文文档

{"query":{"bool":{"must":[{"bool":{"must":[{"exists":{"field":"name"}}],"must_not":[{"wildcard":{"name":{"wildcard":"*"}}}]}}]}}}Spring中文文档

IsNotEmptySpring中文文档

findByNameIsNotEmptySpring中文文档

{"query":{"bool":{"must":[{"wildcard":{"name":{"wildcard":"*"}}}]}}}Spring中文文档

不支持使用方法名称来构建采用参数的 Geo-shape 查询。 如果需要在存储库中具有此类函数,请在自定义存储库实现中使用 with。GeoJsonElasticsearchOperationsCriteriaQuery
表 1.方法名称中支持的关键字
关键词 样本 Elasticsearch 查询字符串

AndSpring中文文档

findByNameAndPriceSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } }, { "query_string" : { "query" : "?", "fields" : [ "price" ] } } ] } }}Spring中文文档

OrSpring中文文档

findByNameOrPriceSpring中文文档

{ "query" : { "bool" : { "should" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } }, { "query_string" : { "query" : "?", "fields" : [ "price" ] } } ] } }}Spring中文文档

IsSpring中文文档

findByNameSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } } ] } }}Spring中文文档

NotSpring中文文档

findByNameNotSpring中文文档

{ "query" : { "bool" : { "must_not" : [ { "query_string" : { "query" : "?", "fields" : [ "name" ] } } ] } }}Spring中文文档

BetweenSpring中文文档

findByPriceBetweenSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : ?, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

LessThanSpring中文文档

findByPriceLessThanSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : null, "to" : ?, "include_lower" : true, "include_upper" : false } } } ] } }}Spring中文文档

LessThanEqualSpring中文文档

findByPriceLessThanEqualSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : null, "to" : ?, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

GreaterThanSpring中文文档

findByPriceGreaterThanSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : null, "include_lower" : false, "include_upper" : true } } } ] } }}Spring中文文档

GreaterThanEqualSpring中文文档

findByPriceGreaterThanEqualSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : null, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

BeforeSpring中文文档

findByPriceBeforeSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : null, "to" : ?, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

AfterSpring中文文档

findByPriceAfterSpring中文文档

{ "query" : { "bool" : { "must" : [ {"range" : {"price" : {"from" : ?, "to" : null, "include_lower" : true, "include_upper" : true } } } ] } }}Spring中文文档

LikeSpring中文文档

findByNameLikeSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?*", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

StartingWithSpring中文文档

findByNameStartingWithSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "?*", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

EndingWithSpring中文文档

findByNameEndingWithSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "*?", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

Contains/ContainingSpring中文文档

findByNameContainingSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "*?*", "fields" : [ "name" ] }, "analyze_wildcard": true } ] } }}Spring中文文档

In(当批注为 FieldType.Keyword 时)Spring中文文档

findByNameIn(Collection<String>names)Spring中文文档

{ "query" : { "bool" : { "must" : [ {"bool" : {"must" : [ {"terms" : {"name" : ["?","?"]}} ] } } ] } }}Spring中文文档

InSpring中文文档

findByNameIn(Collection<String>names)Spring中文文档

{ "query": {"bool": {"must": [{"query_string":{"query": "\"?\" \"?\"", "fields": ["name"]}}]}}}Spring中文文档

NotIn(当批注为 FieldType.Keyword 时)Spring中文文档

findByNameNotIn(Collection<String>names)Spring中文文档

{ "query" : { "bool" : { "must" : [ {"bool" : {"must_not" : [ {"terms" : {"name" : ["?","?"]}} ] } } ] } }}Spring中文文档

NotInSpring中文文档

findByNameNotIn(Collection<String>names)Spring中文文档

{"query": {"bool": {"must": [{"query_string": {"query": "NOT(\"?\" \"?\")", "fields": ["name"]}}]}}}Spring中文文档

TrueSpring中文文档

findByAvailableTrueSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "true", "fields" : [ "available" ] } } ] } }}Spring中文文档

FalseSpring中文文档

findByAvailableFalseSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "false", "fields" : [ "available" ] } } ] } }}Spring中文文档

OrderBySpring中文文档

findByAvailableTrueOrderByNameDescSpring中文文档

{ "query" : { "bool" : { "must" : [ { "query_string" : { "query" : "true", "fields" : [ "available" ] } } ] } }, "sort":[{"name":{"order":"desc"}}] }Spring中文文档

ExistsSpring中文文档

findByNameExistsSpring中文文档

{"query":{"bool":{"must":[{"exists":{"field":"name"}}]}}}Spring中文文档

IsNullSpring中文文档

findByNameIsNullSpring中文文档

{"query":{"bool":{"must_not":[{"exists":{"field":"name"}}]}}}Spring中文文档

IsNotNullSpring中文文档

findByNameIsNotNullSpring中文文档

{"query":{"bool":{"must":[{"exists":{"field":"name"}}]}}}Spring中文文档

IsEmptySpring中文文档

findByNameIsEmptySpring中文文档

{"query":{"bool":{"must":[{"bool":{"must":[{"exists":{"field":"name"}}],"must_not":[{"wildcard":{"name":{"wildcard":"*"}}}]}}]}}}Spring中文文档

IsNotEmptySpring中文文档

findByNameIsNotEmptySpring中文文档

{"query":{"bool":{"must":[{"wildcard":{"name":{"wildcard":"*"}}}]}}}Spring中文文档

不支持使用方法名称来构建采用参数的 Geo-shape 查询。 如果需要在存储库中具有此类函数,请在自定义存储库实现中使用 with。GeoJsonElasticsearchOperationsCriteriaQuery

方法返回类型

可以将存储库方法定义为具有以下返回类型,用于返回多个元素:Spring中文文档

使用@Query注

例 2.使用注释声明对方法的查询。@Query

传递给该方法的参数可以插入到查询字符串的占位符中。占位符的形式为 、 等,表示第一个、第二个、第三个参数等。?0?1?2Spring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("{\"match\": {\"name\": {\"query\": \"?0\"}}}")
    Page<Book> findByName(String name,Pageable pageable);
}

设置为注释参数的 String 必须是有效的 Elasticsearch JSON 查询。 它将作为查询元素的值发送到 Easticsearch;例如,如果使用参数 John 调用函数,它将生成以下查询正文:Spring中文文档

{
  "query": {
    "match": {
      "name": {
        "query": "John"
      }
    }
  }
}
例 3. 对采用 Collection 参数的方法进行注释@Query

存储库方法,例如Spring中文文档

@Query("{\"ids\": {\"values\": ?0 }}")
List<SampleEntity> getByIds(Collection<String> ids);

将进行 ID 查询以返回所有匹配的文档。因此,使用 of 调用该方法将生成查询正文List["id1", "id2", "id3"]Spring中文文档

{
  "query": {
    "ids": {
      "values": ["id1", "id2", "id3"]
    }
  }
}

使用 SpEL 表达式

例 4.使用带有 SpEL 表达式的注释对方法声明查询。@Query

在 中定义查询时,也支持 SpEL 表达式。@QuerySpring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("""
        {
          "bool":{
            "must":[
              {
                "term":{
                  "name": "#{#name}"
                }
              }
            ]
          }
        }
        """)
    Page<Book> findByName(String name, Pageable pageable);
}

例如,如果使用参数 John 调用函数,它将生成以下查询正文:Spring中文文档

{
  "bool":{
    "must":[
      {
        "term":{
          "name": "John"
        }
      }
    ]
  }
}
例 5.访问参数属性。

假设我们有以下类作为查询参数类型:Spring中文文档

public record QueryParameter(String value) {
}

通过符号访问参数很容易,然后使用简单的 :#value.Spring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("""
            {
              "bool":{
                "must":[
                  {
                    "term":{
                      "name": "#{#parameter.value}"
                    }
                  }
                ]
              }
            }
            """)
    Page<Book> findByName(QueryParameter parameter, Pageable pageable);
}

我们现在可以作为参数传递,它将生成与上面相同的查询字符串。new QueryParameter("John")Spring中文文档

例 6.访问 Bean 属性。

还支持访问 Bean 属性。假设有一个名为 type 的 bean,我们可以用 symbol 而不是 来访问这个 bean,并且不需要在查询方法中声明一个类型的参数:queryParameterQueryParameter@#QueryParameterSpring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("""
            {
              "bool":{
                "must":[
                  {
                    "term":{
                      "name": "#{@queryParameter.value}"
                    }
                  }
                ]
              }
            }
            """)
    Page<Book> findByName(Pageable pageable);
}
例 7.SpEL 和参数。Collection

Collection参数也受支持,并且像往常一样易于使用,例如以下查询:StringtermsSpring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("""
            {
              "bool":{
                "must":[
                  {
                    "terms":{
                      "name": #{#names}
                    }
                  }
                ]
              }
            }
            """)
    Page<Book> findByName(Collection<String> names, Pageable pageable);
}
声明 Elasticsearch JSON 查询时,不应引用集合值。

like 的集合将生成以下术语查询:namesList.of("name1", "name2")Spring中文文档

{
  "bool":{
    "must":[
      {
        "terms":{
          "name": ["name1", "name2"]
        }
      }
    ]
  }
}
例 8.参数中的 access 属性。Collection

当参数中的值不是普通的时,SpEL 集合投影可以方便地使用:CollectionStringSpring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("""
            {
              "bool":{
                "must":[
                  {
                    "terms":{
                      "name": #{#parameters.![value]}
                    }
                  }
                ]
              }
            }
            """)
    Page<Book> findByName(Collection<QueryParameter> parameters, Pageable pageable);
}

这会将所有属性值提取为集合中的新值,因此具有与上述相同的效果。valueCollectionQueryParameterSpring中文文档

例 9.使用 ALTER 参数名称@Param

当通过SpEL访问参数时,通过Sping Data中的注释将参数名称更改为另一个参数名称也很有用:@ParamSpring中文文档

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("""
            {
              "bool":{
                "must":[
                  {
                    "terms":{
                      "name": #{#another.![value]}
                    }
                  }
                ]
              }
            }
            """)
    Page<Book> findByName(@Param("another") Collection<QueryParameter> parameters, Pageable pageable);
}
声明 Elasticsearch JSON 查询时,不应引用集合值。