一、概述
分析器(Analyzer)
分析器是包括两个部分:分词器和过滤器。
分词器顾名思意就是将句子分词单个的词
过滤器就是对分词的结果进行筛选,例如中文中将“的”“呀”这些对句子主体意思影响不大的词删除。英语中类似的就是”is”,”a”等等。
通常在索引和分词两端应该采用同样的分词器。solr自带了一些分词器,如果你需要使用自己公司的分词器,那么就需要修改solr模式(Solr schema)。
定义方式
(1)直接定义分词器
<fieldtype name="nametext" class="solr.TextField">
<analyzer class="org.apache.lucene.analysis.WhitespaceAnalyzer"/>
</fieldtype>
(2)定义分词器工厂和相关过滤器
<fieldtype name="text" class="solr.TextField">
<analyzer type="index">
<tokenizer class="solr.StandardTokenizerFactory"/>
<filter class="solr.StandardFilterFactory"/>
<filter class="solr.LowerCaseFilterFactory"/>
</analyzer>
<analyzer type="query">
<tokenizer class="solr.StandardTokenizerFactory"/>
<filter class="solr.StandardFilterFactory"/>
<filter class="solr.LowerCaseFilterFactory"/>
</analyzer>
</fieldtype>
(3)分词器和分词器工厂混合模式
<fieldtype name="text" class="solr.TextField">
<analyzer type="index">
<tokenizer class="solr.StandardTokenizerFactory"/>
<filter class="solr.StandardFilterFactory"/>
<filter class="solr.LowerCaseFilterFactory"/>
</analyzer>
<analyzer type="query" class="org.apache.lucene.analysis.WhitespaceAnalyzer"/>
</fieldtype>
analyzer:在索引(index)单个文档和查询(query)时都被使用。它检查字段的文本并生成一个token流
tokenizers:读取字符流(Reader),把字段数据切分成序列化的Token流。它输出作为第一个Filter的输入(一个Analyzer只有一个Tokenizer)。
filters:检查tokens流,并且保持、转换或抛弃,或生成新的Token流。
二、自定义分词器(可以源码中的WhitespaceTokenizerFactory为例)
1、编写分词器工厂
(1)首先必须继承TokenizerFactory类。如:public class MyVerticalLineTokenizerFactory extends TokenizerFactory{}
(2)编写自定义工厂类的构造方法(必须重写)
public MyVerticalLineTokenizerFactory(Map<String,String> args) {
//args集合即为配置中的相关属性,如<tokenizer useSmart="false" class="org.wltea.analyzer.lucene.IKTokenizerFactory"/>
//则args的key为class和useSmart,super(args)会删除class。
super(args);
rule = get(args, "rule", RULE_NAMES, RULE_JAVA);//从配置中取出rule配置,并将其从args中删除,默认为RULE_NAMES中的RULE_JAVA
if (!args.isEmpty()) {
throw new IllegalArgumentException("Unknown parameters: " + args);
}
}
(3)重写抽象方法。
public Tokenizer create(AttributeFactory factory) {
return new MyVerticalLineTokenizer(factory);
}
2、编写tokenizer
(1)自定义分词器必须继承CharTokenizer类(或者Tokenizer)
(2)重写构造方法
public MyVerticalLineTokenizer(AttributeFactory factory) {
super(factory);
}
(3)重写isTokenChar方法,该方法控制了分词的字符,如下例中以逗号分隔
protected boolean isTokenChar(int c) {
//逗号的ASCII值为44
return !(c == 44);
}
3、编写filter
(1)自定义filter工厂类必须继承TokenFilterFactory类
(2)重写构造方法
public MyTokenizerFilterFactory(Map<String, String> args) {
super(args);
}
(3)重写抽象方法。
public TokenStream create(TokenStream input) {
return new MyTokenizerFilter(input);
}
(4)自定义filter类必须继承TokenFilter
(5)重写构造方法
protected MyTokenizerFilter(TokenStream input) {
super(input);
}
(6)重写抽象方法。
public boolean incrementToken() throws IOException {
if (input.incrementToken()) {
charUtils.toUpperCase(termAtt.buffer(), 0, termAtt.length());
return true;
} else
return false;
}
三、solr官方提供的相关分词器
1.solr.KeywordTokenizerFactory
不管什么内容,整句当成一个关键字
例如: "http://example.com/I-am+example?Text=-Hello" ==> "http://example.com/I-am+example?Text=-Hello"
2.solr.LetterTokenizerFactory
根据字母来分词,抛弃非字母的部分
例如:"I can't" ==> "I", "can", "t"
3.solr.WhitespaceTokenizerFactory
根据空格来分词
例如:"I do" ==> "I", "do"
4.solr.LowerCaseTokenizerFactory
根据字母分词,并将所有字母转换成小写,抛弃非字母的部分
例如:"I can't" ==> "i", "can", "t"
5.solr.StandardTokenizerFactory
分词举例: "I.B.M. cat's can't" ==>
ACRONYM: "I.B.M.", APOSTROPHE:"cat's", APOSTROPHE:"can't"
说明:该分词器,会自动地给每个分词添加type,以便接下来的对type敏感的过滤器进行处理,目前仅仅只有StandardFilter对Token的类型是敏感的。
6.solr.LowerCaseFilterFactory
将分词小写,不处理非单词部分
例如: "I.B.M.", "Solr" ==> "i.b.m.", "solr".