Lucene 代码详解

1.配置开发环境

1.1 Lucene下载

     Lucene是开发全文检索功能的工具包,从官方网站下载Lucene4.10.3,并解压。

     官方网站:http://lucene.apache.org/

     版本:lucene4.10.3

     Jdk要求:1.7以上

     IDE:Eclipse

1.2 使用的jar包

Lucene包:

lucene-core-4.10.3.jar

lucene-analyzers-common-4.10.3.jar

lucene-queryparser-4.10.3.jar

其它:

commons-io-2.4.jar

junit-4.9.jar



2.功能一:创建索引库

     使用indexwriter对象创建索引

2.1 实现步骤

     第一步:创建一个java工程,并导入jar包。

     第二步:创建一个indexwriter对象。

                 ① 指定索引库的存放位置Directory对象

                 ② 指定一个分析器,对文档内容进行分析。

     第三步:创建document对象。

     第四步:创建field对象,将field添加到document对象中。

     第五步:使用indexwriter对象将document对象写入索引库,此过程进行索引创建。并将索引和document对象写入索引库。

     第六步:关闭IndexWriter对象。

2.2 Field域的属性

     是否分析:是否对域的内容进行分词处理。前提是我们要对域的内容进行查询。

     是否索引:将Field分析后的词或整个Field值进行索引,只有索引方可搜索到。

         比如:商品名称、商品简介分析后进行索引,订单号、身份证号不用分析但也要索引,这些将来都要作为查询条件。

     是否存储:将Field值存储在文档中,存储在文档中的Field才可以从Document中获取

         比如:商品名称、订单号,凡是将来要从Document中获取的Field都要存储。

     是否存储的标准:是否要将内容展示给用户

Lucene 代码详解_第1张图片

2.3 代码实现


//创建索引

      @Test

      public void createIndex() throws Exception {



           //指定索引库存放的路径

           //D:\temp\0108\index

           Directory directory = FSDirectory.open(new File("D:\\temp\\0108\\index"));

           //索引库还可以存放到内存中

           //Directory directory = new RAMDirectory();

           //创建一个标准分析器

           Analyzer analyzer = new StandardAnalyzer();

           //创建indexwriterCofig对象

           //第一个参数: Lucene的版本信息,可以选择对应的lucene版本也可以使用LATEST

           //第二根参数:分析器对象

           IndexWriterConfig config = new IndexWriterConfig(Version.LATEST, analyzer);

           //创建indexwriter对象

           IndexWriter indexWriter = new IndexWriter(directory, config);

           //原始文档的路径D:\传智播客\01.课程\04.lucene\01.参考资料\searchsource

           File dir = new File("D:\\传智播客\\01.课程\\04.lucene\\01.参考资料\\searchsource");

           for (File f : dir.listFiles()) {

                 //文件名

                 String fileName = f.getName();

                 //文件内容

                 String fileContent = FileUtils.readFileToString(f);

                 //文件路径

                 String filePath = f.getPath();

                 //文件的大小

                 long fileSize  = FileUtils.sizeOf(f);

                 //创建文件名域

                 //第一个参数:域的名称

                 //第二个参数:域的内容

                 //第三个参数:是否存储

                 Field fileNameField = new TextField("filename", fileName, Store.YES);

                 //文件内容域

                 Field fileContentField = new TextField("content", fileContent, Store.YES);

                 //文件路径域(不分析、不索引、只存储)

                 Field filePathField = new StoredField("path", filePath);

                 //文件大小域

                 Field fileSizeField = new LongField("size", fileSize, Store.YES);



                 //创建document对象

                 Document document = new Document();

                 document.add(fileNameField);

                 document.add(fileContentField);

                 document.add(filePathField);

                 document.add(fileSizeField);

                 //创建索引,并写入索引库

                 indexWriter.addDocument(document);

           }

           //关闭indexwriter

           indexWriter.close();

      }

2.4 使用Luke工具查看索引文件

Lucene 代码详解_第2张图片

 

3.功能二:查询索引

3.1 实现步骤

第一步:创建一个Directory对象,也就是索引库存放的位置。

第二步:创建一个indexReader对象,需要指定Directory对象。

第三步:创建一个indexsearcher对象,需要指定IndexReader对象

第四步:创建一个TermQuery对象,指定查询的域和查询的关键词。

第五步:执行查询。

第六步:返回查询结果。遍历查询结果并输出。

第七步:关闭IndexReader对象

3.2 IndexSearcher搜索方法

Lucene 代码详解_第3张图片

3.3 代码实现


//查询索引库

     @Test

     public void searchIndex() throws Exception {

          //指定索引库存放的路径

          //D:\temp\0108\index

          Directory directory = FSDirectory.open(new File("D:\\temp\\0108\\index"));

          //创建indexReader对象

          IndexReader indexReader = DirectoryReader.open(directory);

          //创建indexsearcher对象

          IndexSearcher indexSearcher = new IndexSearcher(indexReader);

          //创建查询

          Query query = new TermQuery(new Term("filename", "apache"));

          //执行查询

          //第一个参数是查询对象,第二个参数是查询结果返回的最大值

          TopDocs topDocs = indexSearcher.search(query, 10);

          //查询结果的总条数

          System.out.println("查询结果的总条数:"+ topDocs.totalHits);

          //遍历查询结果

          //topDocs.scoreDocs存储了document对象的id

          for (ScoreDoc scoreDoc : topDocs.scoreDocs) {

               //scoreDoc.doc属性就是document对象的id

               //根据document的id找到document对象

               Document document = indexSearcher.doc(scoreDoc.doc);

               System.out.println(document.get("filename"));

               //System.out.println(document.get("content"));

               System.out.println(document.get("path"));

               System.out.println(document.get("size"));

          }

          //关闭indexreader对象

          indexReader.close();

     }

 

4.功能三:支持中文分词

4.1 分析器(Analyzer)的执行过程

     如下图是语汇单元的生成过程:

Lucene 代码详解_第4张图片

     从一个Reader字符流开始,创建一个基于Reader的Tokenizer分词器,经过三个TokenFilter生成语汇单元Tokens。

     要看分析器的分析效果,只需要看Tokenstream中的内容就可以了。每个分析器都有一个方法tokenStream,返回一个tokenStream对象。4.2 分析器的分词效果


//查看标准分析器的分词效果

     public void testTokenStream() throws Exception {

          //创建一个标准分析器对象

          Analyzer analyzer = new StandardAnalyzer();

          //获得tokenStream对象

          //第一个参数:域名,可以随便给一个

          //第二个参数:要分析的文本内容

          TokenStream tokenStream = analyzer.tokenStream("test", "The Spring Framework provides a comprehensive programming and configuration model.");

          //添加一个引用,可以获得每个关键词

          CharTermAttribute charTermAttribute = tokenStream.addAttribute(CharTermAttribute.class);

          //添加一个偏移量的引用,记录了关键词的开始位置以及结束位置

          OffsetAttribute offsetAttribute = tokenStream.addAttribute(OffsetAttribute.class);

          //将指针调整到列表的头部

          tokenStream.reset();

          //遍历关键词列表,通过incrementToken方法判断列表是否结束

          while(tokenStream.incrementToken()) {

               //关键词的起始位置

               System.out.println("start->" + offsetAttribute.startOffset());

               //取关键词

               System.out.println(charTermAttribute);

               //结束位置

               System.out.println("end->" + offsetAttribute.endOffset());

          }

          tokenStream.close();

     }

4.3 中文分析器

4.3.1 Lucene 自带中文分词器

     【1】StandardAnalyzer:

     单字分词:就是按照中文一个字一个字地进行分词。如:“我爱中国”,

     效果:“我”、“爱”、“中”、“国”。

     【2】CJKAnalyzer

     二分法分词:按两个字进行切分。如:“我是中国人”,效果:“我是”、“是中”、“中国”“国人”。

     上边两个分词器无法满足需求。

     【3】SmartChineseAnalyzer

     对中文支持较好,但扩展性差,扩展词库,禁用词库和同义词库等不好处理。

4.3.2 第三方中文分析器

     【1】paoding: 庖丁解牛最新版在?https://code.google.com/p/paoding/?中最多支持Lucene 3.0,且最新提交的代码在 2008-06-03,在svn中最新也是2010年提交,已经过时,不予考虑。

     【2】mmseg4j:最新版已从?https://code.google.com/p/mmseg4j/?移至?https://github.com/chenlb/mmseg4j-solr,支持Lucene 4.10,且在github中最新提交代码是2014年6月,从09年~14年一共有:18个版本,也就是一年几乎有3个大小版本,有较大的活跃度,用了mmseg算法。

     【3】IK-analyzer: 最新版在https://code.google.com/p/ik-analyzer/上,支持Lucene 4.10从2006年12月推出1.0版开始, IKAnalyzer已经推出了4个大版本。最初,它是以开源项目Luence为应用主体的,结合词典分词和文法分析算法的中文分词组件。从3.0版本开 始,IK发展为面向Java的公用分词组件,独立于Lucene项目,同时提供了对Lucene的默认优化实现。在2012版本中,IK实现了简单的分词 歧义排除算法,标志着IK分词器从单纯的词典分词向模拟语义分词衍化。 但是也就是2012年12月后没有在更新,基本无敌了,市场老大。

     【4】ansj_seg:最新版本在?https://github.com/NLPchina/ansj_seg?tags仅有1.1版本,从2012年到2014年更新了大小6次,但是作者本人在2014年10月10日说明:“可能我以后没有精力来维护ansj_seg了”,现在由”nlp_china”管理。2014年11月有更新。并未说明是否支持Lucene,是一个由CRF(条件随机场)算法所做的分词算法。

     【5】imdict-chinese-analyzer:最新版在?https://code.google.com/p/imdict-chinese-analyzer/?, 最新更新也在2009年5月,下载源码,不支持Lucene 4.10 。是利用HMM(隐马尔科夫链)算法。

     【6】Jcseg:最新版本在git.oschina.net/lionsoul/jcseg,支持Lucene 4.10,作者有较高的活跃度。利用mmseg算法。

 

5.功能四:索引库的维护

5.1 索引库的添加

5.1.1 步骤

向索引库中添加document对象。

第一步:先创建一个indexwriter对象

第二步:创建一个document对象

第三步:把document对象写入索引库

第四步:关闭indexwriter。5.1.2 代码实现


//添加索引

     @Test

     public void addDocument() throws Exception {

          //索引库存放路径

          Directory directory = FSDirectory.open(new File("D:\\temp\\0108\\index"));



          IndexWriterConfig config = new IndexWriterConfig(Version.LATEST, new IKAnalyzer());

          //创建一个indexwriter对象

          IndexWriter indexWriter = new IndexWriter(directory, config);

          //创建一个Document对象

          Document document = new Document();

          //向document对象中添加域。

          //不同的document可以有不同的域,同一个document可以有相同的域。

          document.add(new TextField("filename", "新添加的文档", Store.YES));

          document.add(new TextField("content", "新添加的文档的内容", Store.NO));

          document.add(new TextField("content", "新添加的文档的内容第二个content", Store.YES));

          document.add(new TextField("content1", "新添加的文档的内容要能看到", Store.YES));

          //添加文档到索引库

          indexWriter.addDocument(document);

          //关闭indexwriter

          indexWriter.close();

     }

5.2 索引库删除5.2.1 删除全部


//删除全部索引

    @Test

    public void deleteAllIndex() throws Exception {

        IndexWriter indexWriter = getIndexWriter();

        //删除全部索引

        indexWriter.deleteAll();

        //关闭indexwriter

        indexWriter.close();

    }

//说明:将索引目录的索引信息全部删除,直接彻底删除,无法恢复,此方法谨慎使用!  

5.2.2 指定查询条件删除


//根据查询条件删除索引

    @Test

    public void deleteIndexByQuery() throws Exception {

        IndexWriter indexWriter = getIndexWriter();

        //创建一个查询条件

        Query query = new TermQuery(new Term("filename", "apache"));

        //根据查询条件删除

        indexWriter.deleteDocuments(query);

        //关闭indexwriter

        indexWriter.close();

    }

5.2.3 索引库的修改     原理就是先删除后添加。


//修改索引库

    @Test

    public void updateIndex() throws Exception {

        IndexWriter indexWriter = getIndexWriter();

        //创建一个Document对象

        Document document = new Document();

        //向document对象中添加域。

        //不同的document可以有不同的域,同一个document可以有相同的域。

        document.add(new TextField("filename", "要更新的文档", Store.YES));

        document.add(new TextField("content", "2013年11月18日 - Lucene 简介 Lucene 是一个基于 Java 的全文信息检索工具包,它不是一个完整的搜索应用程序,而是为你的应用程序提供索引和搜索功能。", Store.YES));

        indexWriter.updateDocument(new Term("content", "java"), document);

        //关闭indexWriter

        indexWriter.close();

    }

 

6.Lucene索引库查询

    对要搜索的信息创建Query查询对象,Lucene会根据Query查询对象生成最终的查询语法,类似关系数据库Sql语法一样Lucene也有自己的查询语法,比如:“name:lucene”表示查询Field的name为“lucene”的文档信息。

    可通过两种方法创建查询对象:

    (1) 使用Lucene提供Query子类

    Query是一个抽象类,lucene提供了很多查询对象,比如TermQuery项精确查询,NumericRangeQuery数字范围查询等。

    如下代码:

Query query = new TermQuery(new Term("name", "lucene"));

    (2) 使用QueryParse解析查询表达式

    QueryParse会将用户输入的查询表达式解析成Query对象实例。

    如下代码:

QueryParser queryParser = new QueryParser("name", new IKAnalyzer());

Query query = queryParser.parse("name:lucene");

 

6.1 使用query的子类查询

6.1.1 MatchAllDocsQuery

     使用MatchAllDocsQuery查询索引目录中的所有文档

@Test

    public void testMatchAllDocsQuery() throws Exception {

        IndexSearcher indexSearcher = getIndexSearcher();

        //创建查询条件

        Query query = new MatchAllDocsQuery();

        //执行查询

        printResult(query, indexSearcher);

    }

6.1.2 TermQuery

     TermQuery,通过项查询,TermQuery不使用分析器所以建议匹配不分词的Field域查询,比如订单号、分类ID号等。指定要查询的域和要查询的关键词。


//使用Termquery查询

    @Test

    public void testTermQuery() throws Exception {

        IndexSearcher indexSearcher = getIndexSearcher();

        //创建查询对象

        Query query = new TermQuery(new Term("content", "lucene"));

        //执行查询

        TopDocs topDocs = indexSearcher.search(query, 10);

        //共查询到的document个数

        System.out.println("查询结果总数量:" + topDocs.totalHits);

        //遍历查询结果

        for (ScoreDoc scoreDoc : topDocs.scoreDocs) {

            Document document = indexSearcher.doc(scoreDoc.doc);

            System.out.println(document.get("filename"));

            //System.out.println(document.get("content"));

            System.out.println(document.get("path"));

            System.out.println(document.get("size"));

        }

        //关闭indexreader

        indexSearcher.getIndexReader().close();

    }

6.1.3 NumericRangeQuery     可以根据数值范围查询。


//数值范围查询

     @Test

     public void testNumericRangeQuery() throws Exception {

          IndexSearcher indexSearcher = getIndexSearcher();

          //创建查询

          //参数:

          //1.域名

          //2.最小值

          //3.最大值

          //4.是否包含最小值

          //5.是否包含最大值

          Query query = NumericRangeQuery.newLongRange("size", 1l, 1000l, true, true);

          //执行查询

          printResult(query, indexSearcher);

     }

6.1.4 BooleanQuery     可以组合查询条件。


//组合条件查询

     @Test

     public void testBooleanQuery() throws Exception {

          IndexSearcher indexSearcher = getIndexSearcher();

          //创建一个布尔查询对象

          BooleanQuery query = new BooleanQuery();

          //创建第一个查询条件

          Query query1 = new TermQuery(new Term("filename", "apache"));

          Query query2 = new TermQuery(new Term("content", "apache"));

          //组合查询条件

          query.add(query1, Occur.MUST);

          query.add(query2, Occur.MUST);

          //执行查询

          printResult(query, indexSearcher);

     }

说明:Occur.MUST:必须满足此条件,相当于and

           Occur.SHOULD:应该满足,但是不满足也可以,相当于or

          Occur.MUST_NOT:必须不满足。相当于not

 

6.2 使用queryparser查询

     通过QueryParser也可以创建Query,QueryParser提供一个Parse方法,此方法可以直接根据查询语法来查询。Query对象执行的查询语法可通过System.out.println(query)查询,需要使用到分析器。建议创建索引时使用的分析器和查询索引时使用的分析器要一致。

6.2.1 程序实现

@Test

    public void testQueryParser() throws Exception {

        IndexSearcher indexSearcher = getIndexSearcher();

        //创建queryparser对象

        //第一个参数默认搜索的域

        //第二个参数就是分析器对象

        QueryParser queryParser = new QueryParser("content", new IKAnalyzer());

        Query query = queryParser.parse("Lucene是java开发的");

        //执行查询

        printResult(query, indexSearcher);

    }

6.2.2 查询语法


1、基础的查询语法,关键词查询:

域名+“:”+搜索的关键字

例如:content:java

2、范围查询

域名+“:”+[最小值 TO 最大值]

例如:size:[1 TO 1000]

范围查询在lucene中支持数值类型,不支持字符串类型,在solr中支持字符串类型。

3、组合条件查询

1)+条件1 +条件2:两个条件之间是并且的关系and

例如:+filename:apache +content:apache

2)+条件1 条件2:必须满足第一个条件,应该满足第二个条件

例如:+filename:apache content:apache

3)条件1 条件2:两个条件满足其一即可。

例如:filename:apache content:apache

4)条件1 条件2:必须不满足条件1,要满足条件2

6.2.3 MultiFieldQueryParser      可以指定多个默认搜索域


@Test

    public void testMultiFiledQueryParser() throws Exception {

        IndexSearcher indexSearcher = getIndexSearcher();

        //可以指定默认搜索的域是多个

        String[] fields = {"filename", "content"};

        //创建一个MulitFiledQueryParser对象

        MultiFieldQueryParser queryParser = new MultiFieldQueryParser(fields, new IKAnalyzer());

        Query query = queryParser.parse("java AND apache");

        System.out.println(query);

        //执行查询

        printResult(query, indexSearcher);

    }

 

你可能感兴趣的:(ELK)