edismax支持boost函数与score相乘作为,而dismax只能使用bf作用效果是相加,所以在处理多个维度排序时,score其实也应该是其中一个维度 ,用相加的方式处理调整麻烦。

而dismax的实现代码逻辑比较简单,看起来比较易理解,edismax是它的加强版,其实是改变了不少。。比如在以下:

先看看dismax的解析主要实现思路:

首先取出搜索字段名qf

将最终解析成一个BooleanQuery

先解析主mainQuery:

  1. 用户主要是搜索串的解析
  2. altQuery解析处理,看是否使用用户定义的后备搜索串
  3. PhraseQuery解析组装
再解析bq查询,主要是额外加分的查询,不会影响搜索结果数,只会影响排序

再则是bf解析,函数搜索最后会以加的方式作用于文档评分

看主要代码更清晰:

[java] view
plain
copy

  1. @Override
  2. public Query parse() throws ParseException {
  3. SolrParams solrParams = SolrParams.wrapDefaults(localParams, params);
  4. queryFields = SolrPluginUtils.parseFieldBoosts(solrParams.getParams(DisMaxParams.QF));
  5. if (0 == queryFields.size()) {
  6. queryFields.put(req.getSchema().getDefaultSearchFieldName(), 1.0f);
  7. }
  8. /* the main query we will execute.  we disable the coord because
  9. * this query is an artificial construct
  10. */
  11. BooleanQuery query = new BooleanQuery(true);
  12. boolean notBlank = addMainQuery(query, solrParams);
  13. if (!notBlank)
  14. return null;
  15. addBoostQuery(query, solrParams);
  16. addBoostFunctions(query, solrParams);
  17. return query;
  18. }

edismax的主要实现思路跟dismax差不多,以下是一些主要差别之处:

edismax解析含有+,OR,NOT,-语法时,就会忽略掉使用MM。

以下是主要代码实现:

统计搜索串中+,OR ,NOT,-语法元个数

[java] view
plain
copy

  1. // defer escaping and only do if lucene parsing fails, or we need phrases
  2. // parsing fails.  Need to sloppy phrase queries anyway though.
  3. List<Clause> clauses = null;
  4. int numPluses = 0;
  5. int numMinuses = 0;
  6. int numOR = 0;
  7. int numNOT = 0;
  8. clauses = splitIntoClauses(userQuery, false);
  9. for (Clause clause : clauses) {
  10. if (clause.must == '+') numPluses++;
  11. if (clause.must == '-') numMinuses++;
  12. if (clause.isBareWord()) {
  13. String s = clause.val;
  14. if ("OR".equals(s)) {
  15. numOR++;
  16. } else if ("NOT".equals(s)) {
  17. numNOT++;
  18. } else if (lowercaseOperators && "or".equals(s)) {
  19. numOR++;
  20. }
  21. }
  22. }

/////当搜索串里包含有+,OR ,NOT,-这四种时候,mm就会失效

[java] view
plain
copy

  1. boolean doMinMatched = (numOR + numNOT + numPluses + numMinuses) == 0;
  2. (parsedUserQuery != null && doMinMatched) {
  3. String minShouldMatch = solrParams.get(DisMaxParams.MM, "100%");
  4. if (parsedUserQuery instanceof BooleanQuery) {
  5. SolrPluginUtils.setMinShouldMatch((BooleanQuery)parsedUserQuery, minShouldMatch);
  6. }
  7. }

短语查询,先找出普通的查询,原来就是短语查询的、或者属于“OR”,“AND”,“NOT”,’TO‘类型的都不要。由于edismax支持解析符合lucene语法的搜索串,所以不像dismax那样,只需要简单的将搜索串去掉\“,然后加个“”括起来就行

// find non-field clauses

List<Clause>normalClauses =new ArrayList<Clause>(clauses.size());

for (Clauseclause :clauses) {

if (clause.field !=null ||clause.isPhrase)continue;

// check for keywords "AND,OR,TO"

if (clause.isBareWord()) {

String s =clause.val.toString();

// avoid putting explict operators in the phrase query

if ("OR".equals(s) ||"AND".equals(s) ||"NOT".equals(s) || "TO".equals(s))continue;

}

normalClauses.add(clause);

}

// full phrase...

addShingledPhraseQueries(query, normalClauses, phraseFields, 0,

tiebreaker,pslop);

// shingles...

addShingledPhraseQueries(query, normalClauses, phraseFields2, 2,

tiebreaker,pslop);

addShingledPhraseQueries(query, normalClauses, phraseFields3, 3,

tiebreaker,pslop);

////下面是dismax获取短语查询的作法:

[java] view
plain
copy

  1. protected Query getPhraseQuery(String userQuery, SolrPluginUtils.DisjunctionMaxQueryParser pp) throws ParseException {
  2. String userPhraseQuery = userQuery.replace("\"", "");
  3. return pp.parse("\"" + userPhraseQuery + "\"");
  4. }

下面是edismax的作法:

[java] view
plain
copy

  1. private void addShingledPhraseQueries(final BooleanQuery mainQuery,
  2. final List<Clause> clauses,
  3. final Map<String,Float> fields,
  4. int shingleSize,
  5. final float tiebreaker,
  6. final int slop)
  7. throws ParseException {
  8. if (null == fields || fields.isEmpty() ||
  9. null == clauses || clauses.size() <= shingleSize )
  10. return;
  11. if (0 == shingleSize) shingleSize = clauses.size();
  12. final int goat = shingleSize-1; // :TODO: better name for var?
  13. StringBuilder userPhraseQuery = new StringBuilder();
  14. for (int i=0; i < clauses.size() - goat; i++) {
  15. userPhraseQuery.append('"');
  16. for (int j=0; j <= goat; j++) {
  17. userPhraseQuery.append(clauses.get(i + j).val);
  18. userPhraseQuery.append(' ');
  19. }
  20. userPhraseQuery.append('"');
  21. userPhraseQuery.append(' ');
  22. }
  23. ExtendedSolrQueryParser pp =
  24. new ExtendedSolrQueryParser(this, IMPOSSIBLE_FIELD_NAME);
  25. pp.addAlias(IMPOSSIBLE_FIELD_NAME, tiebreaker, fields);
  26. pp.setPhraseSlop(slop);
  27. pp.setRemoveStopFilter(true);  // remove stop filter and keep stopwords
  28. pp.makeDismax = true;
  29. pp.minClauseSize = 2;
  30. Query phrase = pp.parse(userPhraseQuery.toString());
  31. if (phrase != null) {
  32. mainQuery.add(phrase, BooleanClause.Occur.SHOULD);
  33. }
  34. }

edismax技术另一个重要的boost查询,

boost查询也是不会影响搜索结果数,但是影响排序,主要作用是将最后得分以相乘的方式作用于score,函数的解析跟bf差不多。

[java] view
plain
copy

  1. //
  2. // create a boosted query (scores multiplied by boosts)
  3. //
  4. Query topQuery = query;
  5. multBoosts = solrParams.getParams("boost");
  6. if (multBoosts!=null && multBoosts.length>0) {
  7. List<ValueSource> boosts = new ArrayList<ValueSource>();
  8. for (String boostStr : multBoosts) {
  9. if (boostStr==null || boostStr.length()==0) continue;
  10. Query boost = subQuery(boostStr, FunctionQParserPlugin.NAME).getQuery();
  11. ValueSource vs;
  12. if (boost instanceof FunctionQuery) {
  13. vs = ((FunctionQuery)boost).getValueSource();
  14. } else {
  15. vs = new QueryValueSource(boost, 1.0f);
  16. }
  17. boosts.add(vs);
  18. }
  19. if (boosts.size()>1) {
  20. ValueSource prod = new ProductFloatFunction(boosts.toArray(new ValueSource[boosts.size()]));
  21. topQuery = new BoostedQuery(query, prod);
  22. } else if (boosts.size() == 1) {
  23. topQuery = new BoostedQuery(query, boosts.get(0));
  24. }
  25. }

可以看到最后不是一个BooleanQuery,而是一个BoostedQuery。

它就是简单处理子查询的分值再与函数查询的分值相乘返回 :主要的score方法如下:

[java] view
plain
copy

  1. public float score() throws IOException {
  2. float score = qWeight * scorer.score() * vals.floatVal(scorer.docID());
  3. return score>Float.NEGATIVE_INFINITY ? score : -Float.MAX_VALUE;
  4. }

转贴请声明来源:http://blog.csdn.net/duck_genuine/article/details/8060026


solr 的edismax与dismax比较与分析的更多相关文章

  1. solr 使用edismax来控制评分

    如何控制评分 如果设置了sort字段,那么将会按照sort字段的顺序返回结果. 如果没有设置sort字段,那么将会根据相关度打分来排序.也就是说,相关度更高的排在前面. 如何来定制适合自身业务的排序打 ...

  2. solr 请求参数过长报错,Solr配置maxBooleanClauses属性不生效原因分析

    博客分类:   上次已经写过一篇关于solr中,查询条件过多的异常的文章,这次在总结扩展一下: 有时候我们的查询条件会非常多,由于solr的booleanquery默认设置的条件数为1024,所以超过 ...

  3. solr入门之权重排序方法初探之使用edismax改变权重

    做搜索引擎避免不了排序问题,当排序没有要求时,solr有自己的排序打分机制及sorce字段 1.无特殊排序要求时,根据查询相关度来进行排序(solr自身规则) 2.当涉及到一个字段来进行相关度排序时, ...

  4. Solr Dismax查询解析器-深入分析

    Solr 支持多种查询解析,给搜索引擎开发人员提供灵活的查询解析.Solr 中主要包含这几个查询解析器:标准查询解析器.DisMax 查询解析器,扩展 DisMax 查询解析器(eDisMax) Di ...

  5. [solr] - defType - 查询权重排序

    Solr的defType有dismax/edismax两种,这两种的区别,可参见:http://blog.csdn.net/duck_genuine/article/details/8060026 下 ...

  6. solr defType查询权重排序

    Solr的defType有dismax/edismax两种,这两种的区别,可参见:http://blog.csdn.net/duck_genuine/article/details/8060026 下 ...

  7. solr特点三: defType(查询权重排序)

    Solr的defType有dismax/edismax两种,这两种的区别,可参见:http://blog.csdn.net/duck_genuine/article/details/8060026 e ...

  8. 初探solr搜索

    solr是一个基于lucene的搜索引擎,lucene是一个全文检索引擎的架构.solr在此之上进行了封装完善,变成了一个很流行实用的搜索引擎,可以应对绝大部分的搜索需求.使用搜索引擎有以下几点好处: ...

  9. Solr系列五:solr搜索详解(solr搜索流程介绍、查询语法及解析器详解)

    一.solr搜索流程介绍 1. 前面我们已经学习过Lucene搜索的流程,让我们再来回顾一下 流程说明: 首先获取用户输入的查询串,使用查询解析器QueryParser解析查询串生成查询对象Query ...

随机推荐

  1. JS代码执行机制

    JS代码从编译到执行 我们写出一段JS代码,JS的引擎并不是按照我们书写的顺序从上到下顺序编译并且执行的,首先是按照自己的规则对我们的代码先进行编译,然后从上到下执行编译的代码. 在全局作用域中,JS ...

  2. 微信逆向工程之远程操作Mac

    远程控制指令: (功能-指令-是否开启) macbook控制: 屏幕保护-ScreenSave-开启 锁屏-LockScreen-开启 休眠-Sleep-开启 关机-Shutdown-开启 重启-Re ...

  3. [BZOJ5252][八省联考2018]林克卡特树lct

    bzoj(上面可以下数据) luogu description 在树上选出\(k\)条点不相交的链,求最大权值. 一个点也算是一条退化的链,其权值为\(0\). sol 别问我为什么现在才写这题 首先 ...

  4. bzoj2337 XOR和路径

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2337 首先:因为是异或和,所以可以考虑每一位考虑. 就在每一位上求一下该位是1的概率,乘以1 ...

  5. RK3288 device descriptor read/64, error -32

    CPU:RK3288 系统:Android 5.1 主板有两个USB接口,一个接USB摄像头,一个接身份证模块. 插入摄像头可以正常打开,再插入身份证模块时,摄像头就会卡主,而且身份证模块无法识别,内 ...

  6. 【linux】head&&tail

    命令:     head[-n][文件名]                                    tail[-n][-f][文件名] 形式:     -n  行数 (显示前几行)    ...

  7. 【转】纵表、横表互转的SQL

    纵表.横表互转的SQL 原文1:http://takkymj.iteye.com/blog/751401   横表就是普通的建表方式,如一个表结构为: 主键.字段1.字段2.字段3... 如果变成纵表 ...

  8. 不同复制模式下,如何忽略某些binlog事件

    在MySQL复制中,如果slave节点上遇到错误,比如数据不存在或者主键冲突等错误时,想要忽略这些错误,可以采用以下几种方法: 1.未启用GTID模式时 只需通过设定 SQL_SLAVE_SKIP_C ...

  9. MySQL建表规范与常见问题 (go)

    一. 表设计 库名.表名.字段名必须使用小写字母,“_”分割. 库名.表名.字段名必须不超过12个字符. 库名.表名.字段名见名知意,建议使用名词而不是动词. 建议使用InnoDB存储引擎. 存储精确 ...

  10. emacs之配置4,颜色插件

    来自https://github.com/oneKelvinSmith/monokai-emacs/blob/master/monokai-theme.el monokai-theme.el ;;; ...