Lucene全文檢索入門體驗

Lucene是Apache開源的全文檢索框架, 是單純的搜索工具, 簡單易用. 如今已經出到5.2.1的版本, 只需在項目中導入必需的幾個jar包就能使用. 使用的過程能夠歸納爲,java

1)  創建索引網絡

2) 搜索查找, 獲取搜索結果app

 

這裏咱們一塊兒先來學習幾個會用到的核心類:框架


Directory工具

該類在Lucene中用於描述索引存放的位置信息. 好比:學習

 

[java] view plain copy測試

  1. Directory dir = FSDirectory.open(Paths.get("c:\\lucene\\index"));   

其中" c:\\lucene\\index" 是存放索引的文件夾位置.
 搜索引擎

 

Analyzerspa

Analyzer是Lucene的分詞器, 能夠說是分詞解析技術也能夠說是搜索引擎的核心技術之一. 一句話被斷句分詞分析, .使搜索結果更智能更精準. 中文詞庫分詞, 可使用IKAnalyzer等中文分詞工具包..net

 

Analyzer這個類的做用要結合IndexWriterConfig和IndexWriter這兩個類去認識:

IndexWriterConfig, 從類名可知, 是一個保存參數配置的類, 用於生成IndexWriter. 好比:

 

[java] view plain copy

  1. IndexWriterConfig iwc = new IndexWriterConfig(luceneAnalyzer);    
  2.             iwc.setOpenMode(OpenMode.CREATE);    
  3.             IndexWriter indexWriter = new IndexWriter(dir,iwc);   


setOpenMode(...) 設置了IndexWriter的打開方式.

 

固然還有更多的參數設置, 能夠參考這篇文章哦. IndexWriterConfig配置參數說明

 

上面的三行代碼也是建立一個IndexWriter的過程. (dir這個參數就是第一個說起的類Directory.)

IndexWriter 是創建索引的核心類. 若是你也知道Android的SharePreference, SharePreference裏面有一個Editor類. IndexWriter 就是相似Editor這樣的類, 能夠針對索引進行添加(建立新的索引並寫入索引文檔中), 刪除(從索引文檔刪除索引)和更新(更新索引文檔中的索引) 操做. 順便提一下, Lucene的索引, 會生成對應的索引文檔, 因此最好創建文件夾專門存放這些文檔.

 

Document

Document類顧名思義是"文件"類, 其實它是用來存放Field集合. 能夠理解爲存放文件, 通常都是可轉換的文本信息, 好比doc, txt等等. 當把用於被查找的文件信息加入Document後, 再經過

 

[java] view plain copy

  1. indexWriter.addDocument(document);  

這樣就添加了一個索引. 來到這裏, 也許你會想到, 之後要查詢直接先來索引這裏就好了.

 

 

IndexReader

對應於IndexWriter, 就有IndexReader. 建立方法 :

 

[java] view plain copy

  1. IndexReader reader = DirectoryReader.open(FSDirectory.open(Paths.get(index)));  

 

它只讀取索引文檔. 而後交給檢索工具IndexSearcher 去完成查找. 根據傳進Query檢索條件進行檢索查找後, 會獲得一個ScoreDoc類型的結果集, 而後讀取它的Document信息, 就能獲取檢索結果的具體信息, 好比關鍵字包含在哪些內容中, 已經這些內容文檔的存放路徑等等. 這樣就算是完成整個檢索過程了.

 

OK, 下面咱們來簡單的例子體驗下, 有興趣能夠本身去看文檔詳細瞭解哦.

注: 本例子的代碼來自網絡, 是很簡單易懂的例子, 因此就再也不另外寫了, 這裏只是體驗學習, 咱們直接學習別人的. 額, 原做者不知道是誰了, 在這裏感謝一下. 

通過上面的關鍵類的介紹, 相信看下面例子的代碼會容易懂不少了, So 直接上代碼.

1) 創建索引文檔. 

隨便在本地創建一個文件夾, 好比C盤根目錄建立index文件夾. 路徑就是 C:\index . 而要檢索的內容文檔放在 C盤根目錄的source文件夾中, 路徑就是C:\source .

 

[java] view plain copy

  1. public class CreateIndex {  
  2.      public static void main(String[] args) throws Exception {      
  3.          /* 指明要索引文件夾的位置,這裏是C盤的source文件夾下 */      
  4.         File fileDir = new File("C:\\source");      
  5.           
  6.          /* 這裏放索引文件的位置 */      
  7.          //File indexDir = new File("c:\\index");   
  8.          String indexPath = "c:\\index";  
  9.            
  10.          // Directory dir = FSDirectory.open(indexDir);    //v3.6.0  
  11.          Directory dir = FSDirectory.open(Paths.get(indexPath));    
  12.            
  13.          //Analyzer luceneAnalyzer = new StandardAnalyzer(Version.LUCENE_3_6_0);   
  14.          Analyzer luceneAnalyzer = new StandardAnalyzer();  
  15.          IndexWriterConfig iwc = new IndexWriterConfig(luceneAnalyzer);    
  16.          iwc.setOpenMode(OpenMode.CREATE);    
  17.          IndexWriter indexWriter = new IndexWriter(dir,iwc);      
  18.          File[] textFiles = fileDir.listFiles();      
  19.          long startTime = new Date().getTime();      
  20.                
  21.          //增長document到索引去      
  22.          for (int i = 0; i < textFiles.length; i++) {      
  23.              if (textFiles[i].isFile()      
  24.                     ) {      
  25.                  System.out.println("File " + textFiles[i].getCanonicalPath()      
  26.                          + "正在被索引....");      
  27.                  String temp = FileReaderAll(textFiles[i].getCanonicalPath(),      
  28.                          "GBK");      
  29.                  System.out.println(temp);      
  30.                  Document document = new Document();      
  31.                  
  32.                  Field FieldPath = new StringField("path", textFiles[i].getPath(), Field.Store.YES);  
  33.                  Field FieldBody = new TextField("body", temp, Field.Store.YES);      
  34.                  document.add(FieldPath);      
  35.                  document.add(FieldBody);      
  36.                  indexWriter.addDocument(document);      
  37.              }      
  38.          }      
  39.          indexWriter.close();      
  40.                
  41.          //測試一下索引的時間      
  42.          long endTime = new Date().getTime();      
  43.          System.out      
  44.                  .println("這佔用了"      
  45.                          + (endTime - startTime)      
  46.                          + " 毫秒來把文檔增長到索引裏面去!"      
  47.                          + fileDir.getPath());      
  48.      }      
  49.        
  50.      public static String FileReaderAll(String FileName, String charset)      
  51.              throws IOException {      
  52.          BufferedReader reader = new BufferedReader(new InputStreamReader(      
  53.                  new FileInputStream(FileName), charset));      
  54.          String line = new String();      
  55.          String temp = new String();      
  56.                
  57.          while ((line = reader.readLine()) != null) {      
  58.              temp += line;      
  59.          }      
  60.          reader.close();      
  61.          return temp;      
  62.      }      
  63.       
  64. }  


2) 執行檢索查找的類:

 

 

[java] view plain copy

  1. public class ExecuteQuery {  
  2.     public static void main(String[] args) throws  IOException, ParseException  {    
  3.         String index="c:\\index";//搜索的索引路徑    
  4.      //   IndexReader reader=IndexReader.open(FSDirectory.open(Paths.get(index));    //v3.6.0的寫法  
  5.         IndexReader reader = DirectoryReader.open(FSDirectory.open(Paths.get(index)));  
  6.         IndexSearcher searcher=new IndexSearcher(reader);//檢索工具    
  7.         ScoreDoc[] hits=null;    
  8.         String queryString="好";  //搜索的索引名稱    
  9.         Query query=null;    
  10.         Analyzer analyzer= new StandardAnalyzer();    
  11.         try {    
  12.             //QueryParser qp=new QueryParser(Version.LUCENE_3_6_0,"body",analyzer);//用於解析用戶輸入的工具   v3.6.0  
  13.             QueryParser qp=new QueryParser("body",analyzer);//用於解析用戶輸入的工具  
  14.             query=qp.parse(queryString);    
  15.         } catch (ParseException e) {    
  16.             // TODO: handle exception    
  17.         }    
  18.         if (searcher!=null) {    
  19.             TopDocs results=searcher.search(query, 10);//只取排名前十的搜索結果    
  20.             hits=results.scoreDocs;    
  21.             Document document=null;    
  22.             for (int i = 0; i < hits.length; i++) {    
  23.                 document=searcher.doc(hits[i].doc);    
  24.                 String body=document.get("body");    
  25.                 String path=document.get("path");    
  26.                 String modifiedtime=document.get("modifiField");    
  27.                 System.out.println("BODY---"+body+"      ");     
  28.                 System.out.println("PATH--"+path);     
  29.             }    
  30.             if (hits.length>0) {    
  31.                 System.out.println("輸入關鍵字爲:"+queryString+","+"找到"+hits.length+"條結果!");    
  32.                     
  33.             }    
  34.           //  searcher.close();    
  35.             reader.close();    
  36.         }    
  37.     }    
  38.       
  39. }  


例子裏面是搜索"好"字. 效果如圖:

 

而後修改一下內容文檔:

搜索"努力"

中文搜索ok~ 

相關文章
相關標籤/搜索