4、IndexReader和IndexWriter的聲明周期(lucene筆記)

一、說明

在實(shí)際開發(fā)中,由于開啟IndexReaderIndexWriter這兩個(gè)對象很消耗資源,所以一般會使用單例,或者不關(guān)閉而使用一個(gè)池,對應(yīng)于數(shù)據(jù)庫的會話一樣。

二、單例 IndexReader

這里我們還是拷貝工程lucene_index02lucene_index03,在之前工程的基礎(chǔ)上進(jìn)行改進(jìn)。

  • IndexUtil.java類中我們聲明一個(gè)IndexReader的屬性:
private static IndexReader reader = null;//聲明一個(gè)IndexReader的屬性
  • 然后改進(jìn)方法IndexUtil()
public IndexUtil() {
    try {
        setDates();//設(shè)置日期
        scores.put("qq.com", 2.0f);//如果是"qq.com"結(jié)尾的索引則讓其權(quán)值為2.0,注意:默認(rèn)是1.0
        scores.put("sina.edu", 1.5f);
        directory = FSDirectory.open(new File("E:/myeclipse/Lucene/index"));
        reader = IndexReader.open(directory);//在工具一創(chuàng)建的時(shí)候就打開IndexReader
    } catch (IOException e) {
        e.printStackTrace();
    }
}

在構(gòu)造方法中我們實(shí)例化IndexReader,這樣就達(dá)到了單例的效果。

  • 定義一個(gè)方法獲取IndexSearcher類:
public IndexSearcher getSearcher(){
    return new IndexSearcher(reader);
}
  • 改進(jìn)搜索方法:
    public void search02(){
        IndexReader reader;
        try {
            IndexSearcher searcher = getSearcher();
            TermQuery query = new TermQuery(new Term("content", "like"));//搜索內(nèi)容中含有l(wèi)ike的
            TopDocs tds = searcher.search(query, 10);
            for(ScoreDoc sd : tds.scoreDocs){
                Document doc = searcher.doc(sd.doc);
                //這里我們獲取權(quán)值getBoost()的時(shí)候發(fā)現(xiàn)都是1.0,這是因?yàn)檫@里是獲取的一個(gè)document,和原來的沒有關(guān)系。
                //要想看其權(quán)值信息,可以使用luke工具
                //而這里的日期需要我們轉(zhuǎn)換成日期格式
                System.out.println("id號:" + doc.get("id") 
                        + ",權(quán)值:"+ doc.getBoost() 
                        + ",名字:" + doc.get("name") 
                        + ",郵箱:" + doc.get("email")
                        + ",附件條數(shù):" +doc.get("attach") 
                        + ",日期:" + doc.get("date"));
                searcher.close();//注意:這里不是關(guān)閉reader,而是關(guān)閉searcher
            }
        } catch (CorruptIndexException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

注意:這里我們就不要關(guān)閉reader了,而是關(guān)閉searcher,這樣reader就可以一直使用一個(gè)。

  • 測試
    @Test
    public void testSearch02(){
        IndexUtil util = new IndexUtil();
        for(int i = 0; i < 5; i++){
            util.search02();//這里我們讓其執(zhí)行5次
            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

說明:這里我們在測試的時(shí)候先重新生成索引,然后使用上面的測試方法測試,我們每個(gè)10秒執(zhí)行一次搜索,在搜索的過程中,我們再運(yùn)行刪除方法,刪除id為1的索引,但是我們發(fā)現(xiàn)即使刪除了,索引卻并沒有變少,這是因?yàn)槲覀円恢笔褂玫氖峭粋€(gè)reader。但是當(dāng)我們關(guān)閉此測試方法,再使用之前的搜索方法testSearch01測試一下,發(fā)現(xiàn)索引少了一條,針對此問題,我們繼續(xù)改進(jìn)。

  • 改進(jìn)getSearcher方法
public IndexSearcher getSearcher(){
    try {
        if(reader == null){
            reader = IndexReader.open(directory);
        }else{
            //如果reader發(fā)生改變則返回舊的reader,否則創(chuàng)建一個(gè)新的reader
            IndexReader tr = IndexReader.openIfChanged(reader);
            if(tr != null){
                reader.close();//關(guān)閉舊的reader
                reader = tr;
            }
        }
        return new IndexSearcher(reader);
    } catch (CorruptIndexException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

說明:這里我們使用方法openIfChanged,此方法在reader發(fā)生變化的時(shí)候就會返回新的reader,否則返回舊的reader。但是一定注意:我們產(chǎn)生新的reader之后需要將舊的reader關(guān)閉。之后我們在運(yùn)行上面的測試方法過程中使用刪除方法會發(fā)現(xiàn)索引可以實(shí)時(shí)更新了。

三、單例 IndexWriter

對于也是類似,我們拿刪除方法來說明:

    public void delete(){
        IndexWriter writer = null;
        try {
            writer = new IndexWriter(directory, new IndexWriterConfig(Version.LUCENE_35, new StandardAnalyzer(Version.LUCENE_35)));
            
            //參數(shù)可以是一個(gè)選項(xiàng),可以是一個(gè)query,也可以是一個(gè)term,term是一個(gè)精確查找的值
            //這里我們測試此方法之后再次執(zhí)行搜索方法,發(fā)現(xiàn)文檔數(shù)numDocs還有5個(gè),比之前少了一個(gè),但是maxDoc還是6個(gè)
            //在我們的索引目錄中發(fā)現(xiàn)出現(xiàn)了一個(gè)delete的文件。這里的刪除就像一個(gè)回收站一樣,是可以恢復(fù)的
            writer.deleteDocuments(new Term("id", "1"));//這里表示刪除索引為1的id
            writer.commit();
        } catch (CorruptIndexException e) {
            e.printStackTrace();
        } catch (LockObtainFailedException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }/*finally{
            if(writer != null){
                try {
                    writer.close();
                } catch (CorruptIndexException e) {
                    e.printStackTrace();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }*/
    }

這里我們可以不關(guān)閉IndexWriter,但是如果這樣數(shù)據(jù)就不能實(shí)時(shí)更新,此時(shí)我們可以像數(shù)據(jù)庫那樣,提交一下即可。

最后:這里只是簡單演示的了一下單例,但是例子并不完善,比如這里IndexReader確實(shí)是在構(gòu)造函數(shù)中實(shí)例化,是單例的,但是我們在對創(chuàng)建索引方法測試的時(shí)候也會創(chuàng)建一個(gè)IndexReader對象,但是最后卻沒有關(guān)閉,不知道是否是這個(gè)原因?qū)е挛覀儨y試創(chuàng)建對象方法的時(shí)候出現(xiàn)異常,留待后面解決。

具體的單例實(shí)現(xiàn)請參考:http://www.cnblogs.com/likehua/archive/2012/02/20/2359087.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,767評論 18 399
  • //Clojure入門教程: Clojure – Functional Programming for the J...
    葡萄喃喃囈語閱讀 3,772評論 0 7
  • JAVA面試題 1、作用域public,private,protected,以及不寫時(shí)的區(qū)別答:區(qū)別如下:作用域 ...
    JA尐白閱讀 1,183評論 1 0
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,967評論 19 139
  • 如果不是開始記錄,我都不知道原來我已經(jīng)畫了四五本速寫本了,畫了這么多還沒有進(jìn)步,幸好我沒有找老師,否則肯定會敲我腦...
    木子喵0閱讀 474評論 3 1