`

Lucene学习总结之七:Lucene搜索过程解析(2)

阅读更多

二、Lucene搜索详细过程

为了解析Lucene对索引文件搜索的过程,预先写入索引了如下几个文件:

file01.txt: apple apples cat dog

file02.txt: apple boy cat category

file03.txt: apply dog eat etc

file04.txt: apply cat foods

2.1、打开IndexReader指向索引文件夹

代码为:

IndexReader reader = IndexReader.open(FSDirectory.open(indexDir));

其实是调用了DirectoryReader.open(Directory, IndexDeletionPolicy, IndexCommit, boolean, int) 函数,其主要作用是生成一个SegmentInfos.FindSegmentsFile对象,并用它来找到此索引文件中所有的段,并打开这些段。

SegmentInfos.FindSegmentsFile.run(IndexCommit commit)主要做以下事情:

2.1.1、找到最新的segment_N文件

  • 由于segment_N是整个索引中总的元数据,因而正确的选择segment_N更加重要。
  • 然而有时候为了使得索引能够保存在另外的存储系统上,有时候需要用NFS mount一个远程的磁盘来存放索引,然而NFS为了提高性能,在本地有Cache,因而有可能使得此次打开的索引不是另外的writer写入的最新信息,所以在此处用了双保险。
  • 一方面,列出所有的segment_N,并取出其中的最大的N,设为genA

String[] files = directory.listAll();

long genA = getCurrentSegmentGeneration(files);

long getCurrentSegmentGeneration(String[] files) {

    long max = -1;

    for (int i = 0; i < files.length; i++) {

      String file = files[i];

      if (file.startsWith(IndexFileNames.SEGMENTS) //"segments_N"

          && !file.equals(IndexFileNames.SEGMENTS_GEN)) { //"segments.gen"

        long gen = generationFromSegmentsFileName(file);

        if (gen > max) {

          max = gen;

        }

      }

    }

    return max;

  }

  • 另一方面,打开segment.gen文件,从中读出N,设为genB

IndexInput genInput = directory.openInput(IndexFileNames.SEGMENTS_GEN);

int version = genInput.readInt();

long gen0 = genInput.readLong();

long gen1 = genInput.readLong();

if (gen0 == gen1) {

    genB = gen0;

}

  • 在genA和genB中去较大者,为gen,并用此gen构造要打开的segments_N的文件名

if (genA > genB)

    gen = genA;

else

    gen = genB;

String segmentFileName = IndexFileNames.fileNameFromGeneration(IndexFileNames.SEGMENTS, "", gen); //segmentFileName    "segments_4"   

 

2.1.2、通过segment_N文件中保存的各个段的信息打开各个段

  • 从segment_N中读出段的元数据信息,生成SegmentInfos

SegmentInfos infos = new SegmentInfos();

infos.read(directory, segmentFileName);

SegmentInfos.read(Directory, String) 代码如下:

int format = input.readInt();

version = input.readLong();

counter = input.readInt();

for (int i = input.readInt(); i > 0; i—) {

  //读出每一个段,并构造SegmentInfo对象

  add(new SegmentInfo(directory, format, input));

}

 

SegmentInfo(Directory dir, int format, IndexInput input)构造函数如下:

name = input.readString();

docCount = input.readInt();

delGen = input.readLong();

docStoreOffset = input.readInt();

if (docStoreOffset != -1) {

  docStoreSegment = input.readString();

  docStoreIsCompoundFile = (1 == input.readByte());

} else {

  docStoreSegment = name;

  docStoreIsCompoundFile = false;

}

hasSingleNormFile = (1 == input.readByte());

int numNormGen = input.readInt();

normGen = new long[numNormGen];

for(int j=0;j<numNormGen;j++) {

  normGen[j] = input.readLong();

}

isCompoundFile = input.readByte();

delCount = input.readInt();

hasProx = input.readByte() == 1;

其实不用多介绍,看过Lucene学习总结之三:Lucene的索引文件格式 (2)一章,就很容易明白。

  • 根据生成的SegmentInfos打开各个段,并生成ReadOnlyDirectoryReader

SegmentReader[] readers = new SegmentReader[sis.size()];

for (int i = sis.size()-1; i >= 0; i—) {

   //打开每一个段

   readers[i] = SegmentReader.get(readOnly, sis.info(i), termInfosIndexDivisor);

}

SegmentReader.get(boolean, Directory, SegmentInfo, int, boolean, int) 代码如下:

instance.core = new CoreReaders(dir, si, readBufferSize, termInfosIndexDivisor);

instance.core.openDocStores(si); //生成用于读取存储域和词向量的对象。

instance.loadDeletedDocs(); //读取被删除文档(.del)文件

instance.openNorms(instance.core.cfsDir, readBufferSize); //读取标准化因子(.nrm)

CoreReaders(Directory dir, SegmentInfo si, int readBufferSize, int termsIndexDivisor)构造函数代码如下:

cfsReader = new CompoundFileReader(dir, segment + "." + IndexFileNames.COMPOUND_FILE_EXTENSION, readBufferSize); //读取cfs的reader

fieldInfos = new FieldInfos(cfsDir, segment + "." + IndexFileNames.FIELD_INFOS_EXTENSION); //读取段元数据信息(.fnm)

TermInfosReader reader = new TermInfosReader(cfsDir, segment, fieldInfos, readBufferSize, termsIndexDivisor); //用于读取词典信息(.tii .tis)

freqStream = cfsDir.openInput(segment + "." + IndexFileNames.FREQ_EXTENSION, readBufferSize); //用于读取freq

proxStream = cfsDir.openInput(segment + "." + IndexFileNames.PROX_EXTENSION, readBufferSize); //用于读取prox

FieldInfos(Directory d, String name)构造函数如下:

IndexInput input = d.openInput(name);

int firstInt = input.readVInt();

size = input.readVInt();

for (int i = 0; i < size; i++) {

  //读取域名

  String name = StringHelper.intern(input.readString());

  //读取域的各种标志位

  byte bits = input.readByte();

  boolean isIndexed = (bits & IS_INDEXED) != 0;

  boolean storeTermVector = (bits & STORE_TERMVECTOR) != 0;

  boolean storePositionsWithTermVector = (bits & STORE_POSITIONS_WITH_TERMVECTOR) != 0;

  boolean storeOffsetWithTermVector = (bits & STORE_OFFSET_WITH_TERMVECTOR) != 0;

  boolean omitNorms = (bits & OMIT_NORMS) != 0;

  boolean storePayloads = (bits & STORE_PAYLOADS) != 0;

  boolean omitTermFreqAndPositions = (bits & OMIT_TERM_FREQ_AND_POSITIONS) != 0;

  //将读出的域生成FieldInfo对象,加入fieldinfos进行管理

  addInternal(name, isIndexed, storeTermVector, storePositionsWithTermVector, storeOffsetWithTermVector, omitNorms, storePayloads, omitTermFreqAndPositions);

}

CoreReaders.openDocStores(SegmentInfo)主要代码如下:

fieldsReaderOrig = new FieldsReader(storeDir, storesSegment, fieldInfos, readBufferSize, si.getDocStoreOffset(), si.docCount); //用于读取存储域(.fdx, .fdt)

termVectorsReaderOrig = new TermVectorsReader(storeDir, storesSegment, fieldInfos, readBufferSize, si.getDocStoreOffset(), si.docCount); //用于读取词向量(.tvx, .tvd, .tvf)

  • 初始化生成的ReadOnlyDirectoryReader,对打开的多个SegmentReader中的文档编号

 

在Lucene中,每个段中的文档编号都是从0开始的,而一个索引有多个段,需要重新进行编号,于是维护数组start[],来保存每个段的文档号的偏移量,从而第i个段的文档号是从start[i]至start[i]+Num

private void initialize(SegmentReader[] subReaders) {

  this.subReaders = subReaders;

  starts = new int[subReaders.length + 1];

  for (int i = 0; i < subReaders.length; i++) {

    starts[i] = maxDoc;

    maxDoc += subReaders[i].maxDoc();

    if (subReaders[i].hasDeletions())

      hasDeletions = true;

  }

  starts[subReaders.length] = maxDoc;

}

2.1.3、得到的IndexReader对象如下

reader    ReadOnlyDirectoryReader  (id=466)   
    closed    false   
    deletionPolicy    null 

    //索引文件夹  
    directory    SimpleFSDirectory  (id=31)   
        checked    false   
        chunkSize    104857600   
        directory    File  (id=487)   
            path    "D:\\lucene-3.0.0\\TestSearch\\index"   
            prefixLength    3   
        isOpen    true   
        lockFactory    NativeFSLockFactory  (id=488)   
    hasChanges    false   
    hasDeletions    false   
    maxDoc    12   
    normsCache    HashMap<K,V>  (id=483)   
    numDocs    -1   
    readOnly    true   
    refCount    1   
    rollbackHasChanges    false   
    rollbackSegmentInfos    null   

    //段元数据信息
    segmentInfos    SegmentInfos  (id=457)    
        elementCount    3   
        elementData    Object[10]  (id=532)   
            [0]    SegmentInfo  (id=464)   
                delCount    0   
                delGen    -1   
                diagnostics    HashMap<K,V>  (id=537)   
                dir    SimpleFSDirectory  (id=31)   
                docCount    4   
                docStoreIsCompoundFile    false   
                docStoreOffset    -1   
                docStoreSegment    "_0"   
                files    null   
                hasProx    true   
                hasSingleNormFile    true   
                isCompoundFile    1   
                name    "_0"   
                normGen    null   
                preLockless    false   
                sizeInBytes    -1   
            [1]    SegmentInfo  (id=517)   
                delCount    0   
                delGen    -1   
                diagnostics    HashMap<K,V>  (id=542)   
                dir    SimpleFSDirectory  (id=31)   
                docCount    4   
                docStoreIsCompoundFile    false   
                docStoreOffset    -1   
                docStoreSegment    "_1"   
                files    null   
                hasProx    true   
                hasSingleNormFile    true   
                isCompoundFile    1   
                name    "_1"   
                normGen    null   
                preLockless    false   
                sizeInBytes    -1   
            [2]    SegmentInfo  (id=470)   
                delCount    0   
                delGen    -1   
                diagnostics    HashMap<K,V>  (id=547)   
                dir    SimpleFSDirectory  (id=31)   
                docCount    4   
                docStoreIsCompoundFile    false   
                docStoreOffset    -1   
                docStoreSegment    "_2"   
                files    null   
                hasProx    true   
                hasSingleNormFile    true   
                isCompoundFile    1   
                name    "_2"   
                normGen    null   
                preLockless    false   
                sizeInBytes    -1    
        generation    4   
        lastGeneration    4   
        modCount    4   
        pendingSegnOutput    null   
        userData    HashMap<K,V>  (id=533)   
        version    1268193441675   
    segmentInfosStart    null   
    stale    false   
    starts    int[4]  (id=484) 

    //每个段的Reader
    subReaders    SegmentReader[3]  (id=467)   
        [0]    ReadOnlySegmentReader  (id=492)   
            closed    false   
            core    SegmentReader$CoreReaders  (id=495)   
                cfsDir    CompoundFileReader  (id=552)   
                cfsReader    CompoundFileReader  (id=552)   
                dir    SimpleFSDirectory  (id=31)   
                fieldInfos    FieldInfos  (id=553)   
                fieldsReaderOrig    FieldsReader  (id=554)   
                freqStream    CompoundFileReader$CSIndexInput  (id=555)   
                proxStream    CompoundFileReader$CSIndexInput  (id=556)   
                readBufferSize    1024   
                ref    SegmentReader$Ref  (id=557)   
                segment    "_0"   
                storeCFSReader    null   
                termsIndexDivisor    1   
                termVectorsReaderOrig    null   
                tis    TermInfosReader  (id=558)   
                tisNoIndex    null   
            deletedDocs    null   
            deletedDocsDirty    false   
            deletedDocsRef    null   
            fieldsReaderLocal    SegmentReader$FieldsReaderLocal  (id=496)   
            hasChanges    false   
            norms    HashMap<K,V>  (id=500)   
            normsDirty    false   
            pendingDeleteCount    0   
            readBufferSize    1024   
            readOnly    true   
            refCount    1   
            rollbackDeletedDocsDirty    false   
            rollbackHasChanges    false   
            rollbackNormsDirty    false   
            rollbackPendingDeleteCount    0   
            si    SegmentInfo  (id=464)   
            singleNormRef    SegmentReader$Ref  (id=504)   
            singleNormStream    CompoundFileReader$CSIndexInput  (id=506)   
            termVectorsLocal    CloseableThreadLocal<T>  (id=508)   
        [1]    ReadOnlySegmentReader  (id=493)   
            closed    false   
            core    SegmentReader$CoreReaders  (id=511)   
                cfsDir    CompoundFileReader  (id=561)   
                cfsReader    CompoundFileReader  (id=561)   
                dir    SimpleFSDirectory  (id=31)   
                fieldInfos    FieldInfos  (id=562)   
                fieldsReaderOrig    FieldsReader  (id=563)   
                freqStream    CompoundFileReader$CSIndexInput  (id=564)   
                proxStream    CompoundFileReader$CSIndexInput  (id=565)   
                readBufferSize    1024   
                ref    SegmentReader$Ref  (id=566)   
                segment    "_1"   
                storeCFSReader    null   
                termsIndexDivisor    1   
                termVectorsReaderOrig    null   
                tis    TermInfosReader  (id=567)   
                tisNoIndex    null   
            deletedDocs    null   
            deletedDocsDirty    false   
            deletedDocsRef    null   
            fieldsReaderLocal    SegmentReader$FieldsReaderLocal  (id=512)   
            hasChanges    false   
            norms    HashMap<K,V>  (id=514)   
            normsDirty    false   
            pendingDeleteCount    0   
            readBufferSize    1024   
            readOnly    true   
            refCount    1   
            rollbackDeletedDocsDirty    false   
            rollbackHasChanges    false   
            rollbackNormsDirty    false   
            rollbackPendingDeleteCount    0   
            si    SegmentInfo  (id=517)   
            singleNormRef    SegmentReader$Ref  (id=519)   
            singleNormStream    CompoundFileReader$CSIndexInput  (id=520)   
            termVectorsLocal    CloseableThreadLocal<T>  (id=521)   
        [2]    ReadOnlySegmentReader  (id=471)   
            closed    false   
            core    SegmentReader$CoreReaders  (id=475)   
                cfsDir    CompoundFileReader  (id=476)   
                cfsReader    CompoundFileReader  (id=476)   
                dir    SimpleFSDirectory  (id=31)   
                fieldInfos    FieldInfos  (id=480)   
                fieldsReaderOrig    FieldsReader  (id=570)   
                freqStream    CompoundFileReader$CSIndexInput  (id=571)   
                proxStream    CompoundFileReader$CSIndexInput  (id=572)   
                readBufferSize    1024   
                ref    SegmentReader$Ref  (id=573)   
                segment    "_2"   
                storeCFSReader    null   
                termsIndexDivisor    1   
                termVectorsReaderOrig    null   
                tis    TermInfosReader  (id=574)   
                tisNoIndex    null   
            deletedDocs    null   
            deletedDocsDirty    false   
            deletedDocsRef    null   
            fieldsReaderLocal    SegmentReader$FieldsReaderLocal  (id=524)   
            hasChanges    false   
            norms    HashMap<K,V>  (id=525)   
            normsDirty    false   
            pendingDeleteCount    0   
            readBufferSize    1024   
            readOnly    true   
            refCount    1   
            rollbackDeletedDocsDirty    false   
            rollbackHasChanges    false   
            rollbackNormsDirty    false   
            rollbackPendingDeleteCount    0   
            si    SegmentInfo  (id=470)   
            singleNormRef    SegmentReader$Ref  (id=527)   
            singleNormStream    CompoundFileReader$CSIndexInput  (id=528)   
            termVectorsLocal    CloseableThreadLocal<T>  (id=530)   
    synced    HashSet<E>  (id=485)   
    termInfosIndexDivisor    1   
    writeLock    null   
    writer    null   

从上面的过程来看,IndexReader有以下几个特性:

  • 段元数据信息已经被读入到内存中,因而索引文件夹中因为新添加文档而新增加的段对已经打开的reader是不可见的。
  • .del文件已经读入内存,因而其他的reader或者writer删除的文档对打开的reader也是不可见的。
  • 打开的reader已经有inputstream指向cfs文件,从段合并的过程我们知道,一个段文件从生成起就不会改变,新添加的文档都在新的段中,删除的文档都在.del中,段之间的合并是生成新的段,而不会改变旧的段,只不过在段的合并过程中,会将旧的段文件删除,这没有问题,因为从操作系统的角度来讲,一旦一个文件被打开一个inputstream也即打开了一个文件描述符,在内核中,此文件会保持reference count,只要reader还没有关闭,文件描述符还在,文件是不会被删除的,仅仅reference count减一。
  • 以上三点保证了IndexReader的snapshot的性质,也即一个IndexReader打开一个索引,就好像对此索引照了一张像,无论背后索引如何改变,此IndexReader在被重新打开之前,看到的信息总是相同的。
  • 严格的来讲,Lucene的文档号仅仅对打开的某个reader有效,当索引发生了变化,再打开另外一个reader的时候,前面reader的文档0就不一定是后面reader的文档0了,因而我们进行查询的时候,从结果中得到文档号的时候,一定要在reader关闭之前应用,从存储域中得到真正能够唯一标识你的业务逻辑中的文档的信息,如url,md5等等,一旦reader关闭了,则文档号已经无意义,如果用其他的reader查询这些文档号,得到的可能是不期望的文档。

2.2、打开IndexSearcher

代码为:

IndexSearcher searcher = new IndexSearcher(reader);

其过程非常简单:

 

private IndexSearcher(IndexReader r, boolean closeReader) {

  reader = r;

  //当关闭searcher的时候,是否关闭其reader

  this.closeReader = closeReader;

  //对文档号进行编号

  List<IndexReader> subReadersList = new ArrayList<IndexReader>();

  gatherSubReaders(subReadersList, reader);

  subReaders = subReadersList.toArray(new IndexReader[subReadersList.size()]);

  docStarts = new int[subReaders.length];

  int maxDoc = 0;

  for (int i = 0; i < subReaders.length; i++) {

    docStarts[i] = maxDoc;

    maxDoc += subReaders[i].maxDoc();

  }

}

IndexSearcher表面上看起来好像仅仅是reader的一个封装,它的很多函数都是直接调用reader的相应函数,如:int docFreq(Term term),Document doc(int i),int maxDoc()。然而它提供了两个非常重要的函数:

因而在某些应用之中,只想得到某个词的倒排表的时候,最好不要用IndexSearcher,而直接用IndexReader.termDocs(Term term),则省去了打分的计算。

分享到:
评论

相关推荐

    IKAnalyzer中文分词支持lucene6.5.0版本

    由于林良益先生在2012之后未对IKAnalyzer进行更新,后续lucene分词接口发生变化,导致不可使用,所以此jar包支持lucene6.0以上版本

    Lucene学习源码.rar

    通过学习Lucene源码,我们可以定制自己的分词器、查询解析器,甚至优化搜索算法,以满足特定的搜索需求。例如,在中文环境下,可以使用IK Analyzer或者jieba分词库来增强对中文的支持。 总结,Lucene作为Java平台上...

    lucene学习pdf2

    "lucene学习pdf2" 提供的文档,无疑是对Lucene深入理解的一把钥匙,它涵盖了Lucene的核心概念、操作流程以及高级特性。 首先,Lucene的基础知识是必不可少的。Lucene的核心在于索引和搜索,它将非结构化的文本数据...

    lucene学习资料收集

    2. **分词器(Tokenizer)**:Lucene提供了多种分词器,如StandardAnalyzer、ChineseAnalyzer等,用于将输入文本分解成可搜索的词语。 3. **文档索引(Document Indexing)**:索引是Lucene的重要环节,包括创建...

    lucene学习总结

    **Lucene学习总结** 在深入理解Lucene之前,我们首先需要了解什么是全文检索。全文检索是一种从大量文本数据中快速查找所需信息的技术。它通过建立索引来实现高效的搜索,而Lucene正是Java环境下最著名的全文搜索...

    Lucene的的学习资料及案例

    **Lucene学习指南** Lucene是一个高性能、全文检索库,由Apache软件基金会开发并维护,是Java编程语言中广泛使用的搜索引擎库。它提供了一个简单的API,使得开发者能够方便地在应用中实现全文检索功能。本篇文章将...

    Lucene 7.2.1 官方jar包

    总结来说,Lucene 7.2.1 是一个强大的全文检索工具,通过其丰富的功能和高效性能,为开发者提供了构建强大搜索引擎的可能。对于需要处理大量文本数据的应用,使用Lucene进行索引和查询无疑是一个明智的选择。

    Lucene搜索技术

    【Lucene搜索技术】是一种基于Java的全文索引引擎工具包,它并非一个完整的全文搜索引擎,而是提供了一套用于构建全文检索应用的API。Lucene的主要目标是方便开发者将其嵌入到各种应用程序中,实现对特定数据源的...

    lucene 最新版本所有jar包

    同时,它还包含分词器(Analyzer)用于将文本分割成可搜索的词元,以及查询解析器(QueryParser)将用户输入转化为搜索查询。 `lucene-analyzers-common-4.10.2.jar`是Lucene的通用分析器包。分析器是处理文本的...

    Lucene3.3.0学习Demo

    **Lucene 3.3.0 学习Demo** ...总之,"Lucene3.3.0学习Demo"是一个宝贵的资源,对于想要掌握全文搜索技术的开发者来说,它提供了丰富的实践案例和学习材料,可以帮助你快速上手并深入理解Lucene的核心机制。

    lucene学习lucene学习

    Lucene 是一个强大的全文搜索引擎库,它以 Java 语言实现,并作为 Apache 软件基金会的 Apache Jakarta 项目的一部分开放源代码。Lucene 提供了高效、可扩展的索引和搜索功能,允许开发者轻松地在应用程序中集成高级...

    lucene3.6 搜索例子

    《Lucene 3.6 搜索实例解析》 Apache Lucene 是一个开源全文搜索引擎库,为开发者提供了在Java应用程序中实现高效、可扩展的搜索功能的工具。在本篇文章中,我们将深入探讨Lucene 3.6版本中的搜索功能,通过实例...

    lucene学习总结_博客记录1

    本篇文章将深入探讨 Lucene 的核心原理,从全文检索的基础概念出发,逐步解析索引创建过程以及搜索机制。 一、全文检索的基本原理 1. 总论 全文检索是通过索引机制,快速找到文档中包含特定关键词的过程。Lucene ...

    lucene学习资料

    《Lucene学习资料》 Lucene是一个开源的全文搜索引擎库,由Apache软件基金会维护。它提供了高级的文本分析和索引功能,使得开发者能够轻松地在应用程序中集成强大的搜索功能。这个资料包中的《Lucene in Action_2nd...

    经典的lucene实例代码及详细解析以及lucene结构流程介绍

    Lucene应用是指使用Lucene搜索引擎库构建搜索应用程序的过程。Lucene应用程序可以用于各种领域,包括文本搜索、图片搜索和视频搜索等。 在上面的代码中,我们使用了Lucene搜索引擎库构建了一个文本搜索应用程序。该...

    Lucene原理及使用总结

    总的来说,Lucene提供了一套完整的框架,涵盖了从文本处理到搜索结果返回的全过程,使开发者能够专注于构建具有高级搜索功能的应用,而无需关心底层实现细节。通过理解Lucene的基本原理和使用方法,我们可以构建出...

    lucene个人总结

    根据提供的文件信息,以下是对Lucene 3.5版本的核心知识点进行的详细解析与总结: ### Lucene 3.5 概述 Lucene 3.5 是一款高性能的全文检索引擎工具包,广泛应用于搜索引擎、文档管理和内容管理等领域。Lucene 的...

    官方最新完整版lucene-6.6.0.zip

    2. **搜索索引**:使用`IndexSearcher`进行搜索,`QueryParser`帮助解析查询字符串。 3. **读取索引**:`DirectoryReader`用于读取索引,获取文档信息。 4. **评分模型**:通过`Similarity`接口定制评分策略。 5. **...

    Lucene5学习之Group分组统计

    "Lucene5学习之Group分组统计" 这个标题指出我们要讨论的是关于Apache Lucene 5版本中的一个特定功能——Grouping。在信息检索领域,Lucene是一个高性能、全文搜索引擎库,而Grouping是它提供的一种功能,允许用户对...

Global site tag (gtag.js) - Google Analytics