我们正在使用搜索字词字符串构建一个bool查询来搜索我们的Lucene索引。我希望用标准分析仪分析这些字符串,我们用于索引的分析仪。例如,foo-bar 1-2-3
应该被分解为foo
,bar
,1-2-3
,因为Lucene文档声明连字符会导致数字保持在一起,但字词会被标记化。这样做的最佳方式是什么?
目前,我正在通过QueryParser运行搜索字词字符串。
QueryParser parser = new QueryParser("", new StandardAnalyzer());
Query query = parser.parse(aSearchTermString);
这个问题是插入了引号。例如,foo-bar 1-2-3
变为"foo bar"
,1-2-3
,它不返回任何内容,因为Lucene会将foo-bar
标记为foo
和bar
。< / p>
我绝对不想通过删除replace
的引号来破解这种情况,因为我觉得我可能遗漏了某些内容或做错了什么。
答案 0 :(得分:1)
我实际上为StandardAnalyzer
获得了不同的结果。考虑一下这段代码(使用Lucene v4):
public class Tokens {
private static void printTokens(String string, Analyzer analyzer) throws IOException {
System.out.println("Using " + analyzer.getClass().getName());
TokenStream ts = analyzer.tokenStream("default", new StringReader(string));
OffsetAttribute offsetAttribute = ts.addAttribute(OffsetAttribute.class);
CharTermAttribute charTermAttribute = ts.addAttribute(CharTermAttribute.class);
while(ts.incrementToken()) {
int startOffset = offsetAttribute.startOffset();
int endOffset = offsetAttribute.endOffset();
String term = charTermAttribute.toString();
System.out.println(term + " (" + startOffset + " " + endOffset + ")");
}
System.out.println();
}
public static void main(String[] args) throws IOException {
printTokens("foo-bar 1-2-3", new StandardAnalyzer(Version.LUCENE_40));
printTokens("foo-bar 1-2-3", new ClassicAnalyzer(Version.LUCENE_40));
QueryParser standardQP = new QueryParser(Version.LUCENE_40, "", new StandardAnalyzer(Version.LUCENE_40));
BooleanQuery q1 = (BooleanQuery) standardQP.parse("someField:(foo\\-bar\\ 1\\-2\\-3)");
System.out.println(q1.toString() + " # of clauses:" + q1.getClauses().length);
}
}
以上印刷品:
Using org.apache.lucene.analysis.standard.StandardAnalyzer
foo (0 3)
bar (4 7)
1 (8 9)
2 (10 11)
3 (12 13)
Using org.apache.lucene.analysis.standard.ClassicAnalyzer
foo (0 3)
bar (4 7)
1-2-3 (8 13)
someField:foo someField:bar someField:1 someField:2 someField:3 # of clauses:5
因此,上面的代码证明StandardAnalyzer
与ClassicAnalyzer
不同,应该将1-2-3
拆分为不同的令牌 - 完全按照您的意愿。对于查询,您需要转义每个关键字,包括空格,否则QP认为这有不同的含义。
如果您不想转义查询字符串,您可以始终手动对其进行标记(如上面的printTokens
方法),然后用TermQuery
包装每个标记并将所有TermQueries堆叠成一个BooleanQuery
。