解析表达式到lucene.net的Query
查询的时候有自己的查询格式,为了统一并且方便的搜索lucene.net 于是就写了个解析格式,大体上覆盖了几乎所有的lucene.net的query了。当然少了公共扩展库里包含的regexQuery,这个有个坑,平时的时候用的比较少,所以就等下次专门写一篇regexQuery的用法吧。
public class SearchModule
{
private static readonly Regex _all = new Regex(@"^(-?\d+\.\d{1,8}|-?\d+)~(-?\d+\.\d{1,8}|-?\d+)$");
private static readonly Regex _num = new Regex(@"^(-?\d+\.\d{1,8}|-?\d+)$",
RegexOptions.Compiled | RegexOptions.Singleline);
public string Module
{
get;
set;
}
[JsonConverter(typeof(StringEnumConverter))]
public Occur Occur
{
get;
set;
}
private RelationMapper[] _relation;
public RelationMapper[] relation
{
get
{
return this._relation ?? new RelationMapper[] { };
}
set
{
_relation = value;
}
}
private static Query UpQueryMethod(string key, string value)
{
if (_num.IsMatch(value) == false)
throw new Exception("错误的表达式");
else
{
if (value.Contains('.'))
{
return NumericRangeQuery.NewDoubleRange(key, double.Parse(value), double.MaxValue, false, false);
}
else
{
return NumericRangeQuery.NewIntRange(key, int.Parse(value), int.MaxValue, false, false);
}
}
}
private static Query UpEqualQueryMethod(string key, string value)
{
if (_num.IsMatch(value) == false)
throw new Exception("错误的表达式");
else
{
if (value.Contains('.'))
{
return NumericRangeQuery.NewDoubleRange(key, double.Parse(value), double.MaxValue, true, true);
}
else
{
return NumericRangeQuery.NewIntRange(key, int.Parse(value), int.MaxValue, true, true);
}
}
}
private static Query DownQueryMethod(string key, string value)
{
if (_num.IsMatch(value) == false)
throw new Exception("错误的表达式");
else
{
if (value.Contains('.'))
{
return NumericRangeQuery.NewDoubleRange(key, double.MinValue, double.Parse(value), false, false);
}
else
{
return NumericRangeQuery.NewIntRange(key, int.MinValue, int.Parse(value), false, false);
}
}
}
private static Query DownEqualQueryMethod(string key, string value)
{
if (_num.IsMatch(value) == false)
throw new Exception("错误的表达式");
else
{
if (value.Contains('.'))
{
return NumericRangeQuery.NewDoubleRange(key, double.MinValue, double.Parse(value), true, true);
}
else
{
return NumericRangeQuery.NewIntRange(key, int.MinValue, int.Parse(value), true, true);
}
}
}
private static Query RangeQueryMethod(string key, string value)
{
if (_all.IsMatch(value) == false)
throw new Exception("错误的表达式");
else
{
var group = _all.Match(value).Groups;
var startValue = group[1].Value;
var endValue = group[2].Value;
if (startValue.Contains('.'))
{
var sValue = double.Parse(startValue);
var eValue = double.Parse(endValue);
return NumericRangeQuery.NewDoubleRange(key, sValue > eValue ? eValue : sValue, sValue > eValue ? sValue : eValue, true, true);
}
else
{
var sValue = int.Parse(startValue);
var eValue = int.Parse(endValue);
return NumericRangeQuery.NewIntRange(key, sValue > eValue ? eValue : sValue, sValue > eValue ? sValue : eValue, true, true);
}
}
}
private static Query DefaultMethod(string key, string value)
{
return new TermQuery(new Term(key, value));
}
public static Query MultiPhraseQueryMethod(string key, string value)
{
var mulit = new MultiPhraseQuery();
mulit.Add(new Term(key, value));
return mulit;
}
public override string ToString()
{
Func<RelationMapper, Query> queryfunc = (r) =>
{
var c = r.RelationChar;
Func<string, string, Query> method;
switch (c)
{
case ">":
method = (key, value) => UpQueryMethod(key, value);
break;
case ">=":
method = (key, value) => UpEqualQueryMethod(key, value);
break;
case "<":
method = (key, value) => DownQueryMethod(key, value);
break;
case "<=":
method = (key, value) => DownEqualQueryMethod(key, value);
break;
case "=":
method = (key, value) => DefaultMethod(key, value);
break;
case "at":
method = (key, value) => RangeQueryMethod(key, value);
break;
case "contains":
method = (key, value) => MultiPhraseQueryMethod(key, value);
break;
case "like":
method = (key, value) => DefaultMethod(key, value);
break;
default:
throw new Exception("未知的查询表达式");
}
return method(r.Key, r.Value);
};
Func<IEnumerable<RelationMapper>, Query> func = (r) =>
{
var query = new BooleanQuery();
foreach (var item in r)
{
var tquery = queryfunc(item);
if (item.Occur == Occur.AND || item.Occur == Occur.OR)
query.Add(tquery, Lucene.Net.Search.Occur.MUST);
else if (item.Occur == Occur.NOT)
query.Add(tquery, Lucene.Net.Search.Occur.MUST_NOT);
}
return query;
};
var orcount = this.relation.Count(r => r.Occur == Occur.OR);
Query[] result = new Query[orcount + 1];
if (orcount == 0)
{
result[0] = func(this.relation);
}
else
{
var last = 0;
var at = 0;
for (int i = 0; i < this.relation.Length; i++)
{
if (i == 0)
{
continue;
}
if (this.relation[i].Occur == Occur.OR)
{
result[at] = func(this.relation.Take(i).Skip(last));
at++;
last = i;
}
}
if (last != this.relation.Length)
{
result[at] = func(this.relation.Take(this.relation.Length).Skip(last));
}
}
Query resultQuery;
if (result.Count() == 1)
{
resultQuery = result[0];
}
else
{
var bquery = new BooleanQuery();
foreach (var query in result)
{
bquery.Add(query, Lucene.Net.Search.Occur.SHOULD);
}
resultQuery = bquery;
}
return resultQuery.ToString();
}
}
解析表达式到lucene.net的Query的更多相关文章
- Lucene中的 Query对象
"Lucene中的 Query对象": 检 索前,需要对检索字符串进行分析,这是由queryparser来完成的.为了保证查询的正确性,最好用创建索引文件时同样的分析器. quer ...
- 理解Lucene中的Query
Query是一个接口,它有很多实现类. QueryParser是Query解析器,用于将一个字符串解析为一个Query对象,这个Query对象可能属于TermQuery,也可能属于PhraseQuer ...
- Lucene Query Term Weighting
方法 public static Query TermWeighting(Query tquery,Map<String,Float>term2weight){ BooleanQuery ...
- Lucene 06 - 使用Lucene的Query API查询数据
目录 1 Query对象的创建(方式一): 使用子类对象 1.1 常用的Query子类对象 1.2 常用的Query子类对象使用 1.2.1 使用TermQuery 1.2.2 使用NumericRa ...
- Lucene 搜索功能
搜索过程 图解: 主要 API: IndexSearcher: //所有搜索都通过 IndexSearcher 进行,他们将调用该类中重载的 search() 方法 Query: ...
- Lucene.net 多条件查询搜索
最近一直在研究lucene,目的是想让网站实现像搜索引擎那样的搜索,可以快速.准确的帮用户查询出想要的结果.废话不多说,上代码实例: 1.利用BooleanQuery进行多条件搜索(比较灵活) L ...
- 初识Lucene.net
最近想提高下自己的能力,也是由于自己的项目中需要用到Lucene,所以开始接触这门富有挑战又充满新奇的技术.. 刚刚开始,只是写了个小小的demo,用了用lucene,确实很好 创建索引 Data ...
- org.apache.lucene.queryParser.ParseException: Encountered "<EOF>" at line 1, column 0.
如果出现了下列错误,那是因为用错了函数.把queryParser.Query改称queryParser.parse就通过了 org.apache.lucene.queryParser.ParseExc ...
- lucene&solr-day1
全文检索课程 Lucene&Solr(1) 1. 计划 第一天:Lucene的基础知识 1.案例分析:什么是全文检索,如何实现全文检索 2.Lucene实现全文检索的流程 a) ...
随机推荐
- CSS中一些不经意的细节问题1
CSS这样的语法,细节问题非常多,往往一些难以处理的问题,有可能是一些细节问题不到位,所以先记下一些,留给以后自己看看. 1.line-height:150%与line-height:1.5 的区别 ...
- C语言实现二叉树-02版
---恢复内容开始--- 昨天,提交完我们的二叉树项目后,今天早上项目经理早早给我打电话: 他说,小伙子干的不错.但是为什么你上面的insert是recusive的呢? 你难道不知道万一数据量大啦!那 ...
- Linux操作系统shell与函数详解
shell和函数的定义 1. linux shell 函数 将一组命令集或语句形成一个可用的块, 这些语句块称为函数. 2. shell 函数的组成 函数名:函数名字,注意一个脚本中函数名要唯一, ...
- ucos操作系统的内核有哪些调度方法
1)时间片轮番调度法 假设系统中有5个任务,T1,T2,T3,T4,T5,这个时候,操作系统为每一个任务分配时间,比如说我们为T1任务分配10毫秒,为T2任务分配20毫秒,为T3任务分配5毫秒,为T4 ...
- Ninja Blocks物联网平台简介
Ninja Blocks是一个物联网控制平台,其平台架构包括硬件层.处理器层.软件层以及平台层,请看下图: 最底层是硬件层,包括传感器(Sensors)和驱动器(Actuators),例如温度传感器. ...
- 远哥教你MuleESB系列视频教程
远哥教你MuleESB系列视频课程介绍(共11个视频) 1.Mule ESB介绍 2.社区版/企业版的区别和安装 3.MuleESB快速入门,以及MEL和Message结构 4.官方例子讲解( ...
- linux内核设计模式
原文来自:http://lwn.net/Articles/336224/ 选择感兴趣内容简单翻译了下: 在内核社区一直以来的兴趣是保证质量.我们需要保证和改善质量是显而易见的.但是如何做到却不是那么简 ...
- <后会无期>经典影评
先说明是转载,任何不同意见请对原作者表达,楼主不作任何回应,楼主影商极低,楼主觉得这二十几年来看的最好的电影是<一代宗师>,楼主只是觉得这篇影评精彩才发布上来让更多的人看到.原作者意见和楼 ...
- PHP查看SSL证书信息
<? $str = file_get_contents('2.cer'); print_r(openssl_x509_parse($str)); ?> 证书需要使用base64编码的方式c ...
- 在JS中设置Select和radio选中
<select id="Gender" name="Gender"> <option value="1">男< ...