今天我们要使用 Lucene 来实现一个简单的搜索引擎,我们要使用上一节爬取的果壳网语料库来构建索引,然后在索引的基础上进行关键词查询。

    上一节果壳网的语料库放在了 Redis 中如下,有一个有效的文章 ID 集合,对于每一篇文章都会有一个 hash 结构存储了它的标题和 HTML 内容。

    1. valid_article_ids => set(article_id)
    2. article_${id} => hash(title=>${title}, html=>${html})

    种不同功用的文件。构建索引的目标就是生成倒排索引,在本例中,会建立一个 title 标题的倒排索引和一个 html 内容的倒排索引,这是两个不同的倒排索引。

    倒排索引就是分词词汇和文档 ID 列表的映射。如果是英文,那么就是一个个的英文单词,如果是中文,就需要中文分词词库来切分标题和内容来得到一个个的中文词语。这里的「文档 ID」 不是指文章 ID,而是 Lucene 内部的 Document 对象的唯一 ID。我们通过调用 Lucene 的 addDocument 方法添加进去的每一篇文章在 Lucene 内部都会有一个 Document 对象。

    1. class InvertedIndex {
    2. Map<String, int[]> mappings; // word => docIds
    3. }
    4. class Documents {
    5. Map<int, Document> docs; // docId => document
    6. }

    在 Lucene 中,文档 ID 是一个 32bit 的「有符号整数」,按顺序添加进来的文档其 ID 也是连续递增的。因为它是 32bit,这也意味着 Lucene 的单个索引最多能存储 1<<31 -1 篇文档。文档 ID 之间可能会有空隙,因为 Lucene 的文档是支持删除操作的。

    Elasticsearch 为了支持海量的文档存储,它内部对索引进行了分片存储(Sharding)。在内部实现中,它会使用到多个 Lucene 的索引来聚合处理。

    好,下面我们来看看文档索引的构建是如何使用代码来完成的。首先导入 Lucene 依赖库

    1. <dependency>
    2. <groupId>org.apache.lucene</groupId>
    3. <artifactId>lucene-core</artifactId>
    4. <version>8.2.0</version>
    5. </dependency>
    6. <dependency>
    7. <groupId>com.hankcs.nlp</groupId>
    8. <artifactId>hanlp-lucene-plugin</artifactId>
    9. <version>1.1.6</version>
    10. </dependency>
    1. // 中文分词分析器
    2. var analyser = new HanLPAnalyzer();
    3. // 指定索引的存储目录
    4. var directory = FSDirectory.open(Path.of("./guokr"));
    5. // 构造配置对象
    6. var config = new IndexWriterConfig(analyser);
    7. // 构造 IndexWriter 对象
    8. var indexWriter = new IndexWriter(directory, config);
    1. var redis = new JedisPool();
    2. // 首先拿到所有的文章ID
    3. var db = redis.getResource();
    4. var articleIds = db.smembers("valid_article_ids");
    5. db.close();
    6. // 挨个添加
    7. for(var id : articleIds) {
    8. // 取文章的标题和内容
    9. db = redis.getResource();
    10. var key = String.format("article_%s", id);
    11. var title = db.hget(key, "title");
    12. var html = db.hget(key, "html");
    13. var url = String.format("https://www.guokr.com/article/%s/", id);
    14. db.close();
    15. if(title != null && html != null) {
    16. // 干掉内容中所有的 HTML 标签只剩下纯文本
    17. var content = Jsoup.parse(html).text();
    18. // 构造文档
    19. var doc = new Document();
    20. doc.add(new TextField("title", title, Field.Store.YES));
    21. doc.add(new TextField("content", content, Field.Store.YES));
    22. doc.add(new StoredField("url", url));
    23. // 这里添加文档
    24. indexWriter.addDocument(doc);
    25. }
    26. }
    1. indexWriter.close();
    2. directory.close();

    上面的代码最关键的地方在于 Document 对象的构造

    1. var doc = new Document();
    2. doc.add(new TextField("title", title, Field.Store.YES));
    3. doc.add(new TextField("content", content, Field.Store.YES));
    4. doc.add(new StoredField("url", url));

    注意到 TextField 对象的最后一个参数指明是否存储字段的内容,如果这个字段设置为 Field.Store.NO,那么 Lucene 就不存储这个字段的值,但是还是会将这个值的文本进行切词后放入倒排索引中。在关键词查询阶段,我们可以根据关键词搜索到文档 ID,进一步得到这个文档的具体内容,但是文档的内容会缺失这个字段,因为 Lucene 没有存它。简单的说这个字段是隐身的,它在搜索时会起到作用,但是最终的搜索结果里却看不见它。之所以提供这个选项,很明显这是为了可以节约存储空间。

    同时我们还注意到 url 字段使用了 StoreField,这是啥意思?它的意思和 Field.Store.NO 正好相反。它只存储字段的值,不参与检索,相当于文档的附加字段。通俗点讲它就是个「搭便车」字段 —— 老司机带带我。

    现在让我们跑一跑这个程序,跑完之后打开 ./guokr 目录,看看里面都有些啥。

    1. bash> ls -l
    2. total 13824
    3. -rw-r--r-- 1 qianwenpin staff 299B 9 4 14:22 _0.cfe
    4. -rw-r--r-- 1 qianwenpin staff 6.7M 9 4 14:22 _0.cfs
    5. -rw-r--r-- 1 qianwenpin staff 383B 9 4 14:22 _0.si
    6. -rw-r--r-- 1 qianwenpin staff 137B 9 4 14:22 segments_1
    7. -rw-r--r-- 1 qianwenpin staff 0B 9 4 14:22 write.lock

    image.png
    Lucene 虽然不允许多进程同时写,但是可以单进程写多进程读,也就是单写多读。好接下来我们开始尝试 Lucene 的读操作 —— 关键词查询。

    查询操作需要构造一个关键对象 IndexSearcher,它的构造方式比 IndexWriter 简单很多。

    var directory = FSDirectory.open(Path.of("./guokr"));
    var reader = DirectoryReader.open(directory);
    var searcher = new IndexSearcher(reader);
    
    var query = new TermQuery(new Term("title", "动物"));
    
    var hits = searcher.search(query, 10).scoreDocs;
    for(var hit : hits) {
        var doc = searcher.doc(hit.doc);
        System.out.printf("%s=>%s\n", doc.get("url"), doc.get("title"));
    }
    
    reader.close();
    directory.close();
    

    image.png
    所有的文章标题里确实都有「动物」这个词。下面我们改变一下查询的输入,改为从内容查询,并且必须同时包含「动物」和 「世界」两个词汇。这是一个复合查询,复合查询需要使用到一个关键的类 BooleanQuery,它可以对多个子 Query 进行逻辑组合来融合查询结果。

    var query1 = new TermQuery(new Term("content", "动物"));
    var query2 = new TermQuery(new Term("content", "世界"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.MUST)
            .add(query2, BooleanClause.Occur.MUST)
            .build();
    

    image.png
    我们可以点开链接看看文章的内容进行验证一下。
    image.png
    下面我们继续改变查询条件,还是从内容查询,但是条件变为包含「动物」但是不得有「世界」这个词汇,估计满足这样条件的文章会非常多。

    var query1 = new TermQuery(new Term("content", "动物"));
    var query2 = new TermQuery(new Term("content", "世界"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.MUST)
            .add(query2, BooleanClause.Occur.MUST_NOT)
            .build();
    
    var query1 = new TermQuery(new Term("content", "动物"));
    var query2 = new TermQuery(new Term("content", "经济"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.SHOULD)
            .add(query2, BooleanClause.Occur.SHOULD)
            .build();
    
    var query1 = new TermQuery(new Term("content", "动物"));
    var query2 = new TermQuery(new Term("content", "经济"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.MUST)
            .add(query2, BooleanClause.Occur.SHOULD)
            .build();
    

    前面提到 MUST 表示必须包含,MUST_NOT 表示必须不包含。但是如果将两个 MUST_NOT 组合你得到的将会是空查询。为什么会这样呢?

    var query1 = new TermQuery(new Term("content", "动物"));
    var query2 = new TermQuery(new Term("content", "经济"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.MUST_NOT)
            .add(query2, BooleanClause.Occur.MUST_NOT)
            .build();
    
    var query1 = new TermQuery(new Term("content", "动物"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.MUST_NOT)
            .build();
    

    最后我们再看一下 FILTER 选项的作用,它和 SHOULD 正好相反。SHOULD 不影响查询结果,但是会影响排序,而 FILTER 会影响查询结果但是不影响排序,它只起到过滤的作用,就好比数据库查询里的 Where 条件。

    var query1 = new TermQuery(new Term("content", "动物"));
    var query2 = new TermQuery(new Term("content", "经济"));
    var query = new BooleanQuery.Builder()
            .add(query1, BooleanClause.Occur.FILTER)
            .add(query2, BooleanClause.Occur.FILTER)
            .build();
    

    关于 Lucene 查询语句的更多奥秘,在后面的文章中我们会继续深入探讨。


    原文链接:https://blog.csdn.net/shellquery/article/details/100893038