7

注意: 我最初发布这个问题的方式有点不同,不值得更新,因为阅读后我学到了更多。

要求

搜索文档并根据文档中的嵌套元素计算自定义分数。

结构

{
  "mappings": {
    "book": {
      "properties": {
        "title":        { "type": "string", "index": "not_analyzed" },
        "topics": {
          "type": "nested",
          "properties": {
            "title":   { "type": "string", "index": "not_analyzed" },
            "weight":  { "type": "int" }
          }
        }
      }
    }
  }
}

示例查询

{
  "query": {
    "function_score": {
      "query": {
        "term": { "title": "The Magical World of Spittle" }
      },
      "script_score": {
        "script": {
          "lang": "painless",
          "inline": "int score = 0; for(int i = 0; i < doc['topics'].values.length; i++) { score += doc['topics'][i].weight; } return score;",
          "params": {
            "default_return_value": 100
          }
        }
      }
    }
  }
}

孤立无痛

int score = 0;
for(int i = 0; i < doc['topics'].values.length; i++) {
  score += doc['topics'][i].weight;
}
return score;

错误

在类型为 [book] 的映射中未找到 [topics] 的字段

问题

  • 怎么了?
  • 该怎么办?
4

1 回答 1

12

嵌套文档存储在索引中的不同文档中,因此您无法通过父文档中的 doc 值访问它们。您需要使用源文档并导航到该topics.weight属性,如下所示:

隔离无痛:

int score = 0; 
for(int i = 0; i < params._source['topics'].size(); i++) { 
    score += params._source['topics'][i].weight; 
}
return score;

完整查询:

{
  "query": {
    "function_score": {
      "query": {
        "term": { "title": "Book 1" }
      },
      "script_score": {
        "script": {
          "lang": "painless",
          "inline": "int score = 0; for(int i = 0; i < params._source['topics'].size(); i++) { score += params._source['topics'][i].weight; } return score;",
          "params": {
            "default_return_value": 100
          }
        }
      }
    }
  }
}

PS:还要注意类型int不存在,它是integer

于 2017-10-23T04:34:33.780 回答