ES exists不生效

366 阅读1分钟

前提

判断 ES 字段是否有值时,使用到了 exists 语句判断字段是否有值,但是未生效。

我的情况

检查发现字段类型如下,是 nested 类型的,所以不能使用常规的方式查询

在这里插入图片描述

解决方式

如果你的字段也是 nested 的,那么可以参考如下方式。

{
    "query": {
        "bool": {
            "filter": [
            	// 省略其他查询条件
                {
                    "nested": {
                        "query": {
                            "bool": {
                                "should": [
                                    
                                    {
                                        "exists": {
                                        	// 希望不为空的字段
                                            "field": "punished_parties.testaaa",
                                            "boost": 1.0
                                        }
                                    }
                                ],
                                "adjust_pure_negative": true,
                                "boost": 1.0
                            }
                        },
                        // 填写该nested字段的名称
                        "path": "punished_parties",
                        "ignore_unmapped": false,
                        // 是否要用来计算得分,我这里不用
                        "score_mode": "none",
                        "boost": 1.0
                    }
                }
            ]
        }
    }
}

原理

ES 存储对象时,将对象扁平化处理。如下

PUT user/001
{
  "user_name" : [ 
    {
      "first" : "王",
      "last" :  "五"
    },
    {
      "first" : "赵",
      "last" :  "四"
    }
  ]
}

会被存储为

user_name.first = ["王", "赵"]
user_name.last = ["五", "四"]

想要让整体被当作一个对象查询和处理,就需要使用 nested,ES底层将其处理为一个单独索引,这样就会将其视为一个对象整体进行查询和处理了。

参考nested原理和应用文章

为什么使用nested1

为什么使用nested2

nested讲解-1

nested讲解-Object Fields VS. Nested Field Types in Elasticsearch