`
chen_sheng_lin
  • 浏览: 5672 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
社区版块
存档分类
最新评论

全文检索lucene

阅读更多
一、下载lucene4.7的jar包:
lucene-analyzers-common-4.7.0.jar
lucene-analyzers-smartcn-4.7.0.jar
lucene-core-4.7.0.jar
lucene-facet-4.7.0.jar
lucene-highlighter-4.7.0.jar
lucene-queries-4.7.0.jar
lucene-queryparser-4.7.0.jar

把以上jar包导入项目中

二、创建索引
*************************************************1.创建索引配置****************************************
因为创建索引是针对表的,所以定义配置文件,配置需要创建索引的SQL
index.xml:

<?xml version='1.0' encoding='UTF-8'?>
<indexs>
  <index>
    <name>riskRule</name>
    <all>
      <![CDATA[
        select ID,NAME,BODY,DOCUMENT_TYPE from ARMS.T_RISK_RULES
        where remove_flag = 0
        ORDER BY ID
      ]]>
    </all>
    <add>
      <![CDATA[
        SELECT ID,NAME,BODY,DOCUMENT_TYPE from ARMS.T_RISK_RULES
        WHERE remove_flag = 0
        and  ID > {?#ID#}
        AND UPDATE_TIMESTAMP > {?#UPDATE_TIME#}
      ]]>
    </add>
    <update>
      <![CDATA[
        SELECT ID,NAME,BODY,DOCUMENT_TYPE from ARMS.T_RISK_RULES
        WHERE remove_flag = 0
        and ID < {?#ID#}
        AND UPDATE_TIMESTAMP > {?#UPDATE_TIME#}
      ]]>
    </update>
    <delete>
      <![CDATA[
        SELECT ID,NAME,BODY,DOCUMENT_TYPE from ARMS.T_RISK_RULES
        WHERE remove_flag = 1
        AND ID < {?#ID#}
        AND UPDATE_TIMESTAMP > {?#UPDATE_TIME#}
      ]]>
    </delete>
    <blob>BODY:DOCUMENT_TYPE</blob>
  </index>
    <index>
        <name>riskProblem</name>
        <all>
            <![CDATA[
          SELECT ID,TITLE,CONTENTS,PUNISH,CRITERION_CONTENT FROM ARMS.T_RISK
          WHERE REMOVE_FLAG = 0
          ORDER BY ID
        ]]>
        </all>
        <add>
            <![CDATA[
          SELECT ID,TITLE,CONTENTS,PUNISH,CRITERION_CONTENT FROM ARMS.T_RISK
          WHERE REMOVE_FLAG = 0
          AND ID > {?#ID#}
          AND UPDATE_TIME > {?#UPDATE_TIME#}
        ]]>
        </add>
        <update>
            <![CDATA[
          SELECT ID,TITLE,CONTENTS,PUNISH,CRITERION_CONTENT FROM ARMS.T_RISK
          WHERE REMOVE_FLAG = 0
          AND ID < {?#ID#}
          AND UPDATE_TIME > {?#UPDATE_TIME#}
        ]]>
        </update>
        <delete>
        <![CDATA[
          SELECT ID,TITLE,CONTENTS,PUNISH,CRITERION_CONTENT FROM ARMS.T_RISK
          WHERE REMOVE_FLAG = 1
          AND ID < {?#ID#}
          AND UPDATE_TIME > {?#UPDATE_TIME#}
        ]]>
        </delete>
        <blob></blob>
    </index>
</indexs>


读取解析index.xml的工具类
package com.lhzq.ibms.lucene.util;

import com.htsc.abms.lucene.model.Index;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.w3c.dom.Document;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import java.io.File;
import java.util.ArrayList;
import java.util.List;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-10
* Time: 上午9:28
* 索引配置解析
*/
public class IndexConfigMgr
{
    /**
     *  自身对象用来做单例
     */
    private static IndexConfigMgr indexConfigMgr;

    /**
     * 用来做线程锁
     */
    private static Object obj = new Object();

    /**
     *  要索引的表的配置
     */
    private List<Index> tableConfigs;

    /**
     * 日志
     */
    private static Logger logger= LoggerFactory.getLogger(IndexConfigMgr.class);

    /**
     * 索引配置文件
     */
    private static final String INDEX_DIR = "index/index.xml";

    /**
     * index节点的名称
     */
    private static final String INDEX_NODE_NAME = "index";

    /**
     * name节点的名称
     */
    private static final String NAME_NODE_NAME = "name";

    /**
     * all节点的名称
     */
    private static final String ALL_NODE_NAME = "all";

    /**
     * add节点的名称
     */
    private static final String ADD_NODE_NAME = "add";

    /**
     * update节点的名称
     */
    private static final String UPDATE_NODE_NAME = "update";

    /**
     * delete节点的名称
     */
    private static final String DELETE_NODE_NAME = "delete";

    /**
     * blob节点的名称
     */
    private static final String BLOB_NODE_NAME = "blob";

    /**
     * 私有的构造方法
     */
    private  IndexConfigMgr()
    {
        // 创建配置容器
        tableConfigs = new ArrayList<Index>();
    }

    /**
     * 获取实例对象
     * @return
     */
    public static IndexConfigMgr getInstance()
    {
         synchronized (obj)
         {
             if(null == indexConfigMgr)
             {
                 indexConfigMgr = new IndexConfigMgr();
             }
         }

        // 加载配置文件
        indexConfigMgr.load();

        return  indexConfigMgr;
    }

    /**
     * 加载配置文件
     */
    private void load()
    {
        // 拿到索引配置文件的路径
        String path = WorkSpaceCenter.getClassPath(INDEX_DIR);

        Document doc = null;
        try
        {
            doc = getDocumentByPath(path);
            loadIndexes(doc);
        } catch (Exception e) {
            logger.error("加载index.xml文件失败",e);
        }
    }

    /**
     *  根据xml文件路径拿到dom对象
     * @param path 文件路径
     * @return
     * @throws javax.xml.parsers.ParserConfigurationException
     * @throws java.io.IOException
     * @throws org.xml.sax.SAXException
     */
    private Document getDocumentByPath(String path) throws Exception
    {
       // 获取DOM解析器工厂对象
       DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();

       // 获取DOM解析器对象
       DocumentBuilder db = dbf.newDocumentBuilder();

        File file=new File(path);

        // 加载要解析xml文档
       Document doc = db.parse(file);

      return doc;
    }


    /**
     *  加载索引配置
     * @param doc
     * @return
     * @throws javax.xml.parsers.ParserConfigurationException
     * @throws java.io.IOException
     * @throws org.xml.sax.SAXException
     */
    private void loadIndexes(Document doc)
    {
       NodeList indexNodes = doc.getElementsByTagName(INDEX_NODE_NAME);

       Node node = null;
       tableConfigs.clear();
       for (int i = 0; i < indexNodes.getLength() ; i++)
       {
           node = indexNodes.item(i);
           if(!node.hasChildNodes())
           {
               continue;
           }

           tableConfigs.add(newIndex(node));
       }
    }


    /**
     * 封装一个index
     * @param parent
     * @return
     */
    private Index newIndex(Node parent)
    {
        Node node= null;
        Index index = null;

        String name = null;
        String all = null;
        String add = null;
        String update = null;
        String delete = null;
        String blob = null;

        NodeList nodes = parent.getChildNodes();
        for(int i = 0; i < nodes.getLength(); i++)
        {
             node =  nodes.item(i);

            if(!node.hasChildNodes())
            {
                continue;
            }

            if(node.getNodeName().equals(NAME_NODE_NAME))
            {
                name = node.getTextContent().trim();
            }

            if(node.getNodeName().equals(ALL_NODE_NAME))
            {
                all = node.getTextContent().trim();
            }

            if(node.getNodeName().equals(ADD_NODE_NAME))
            {
                add = node.getTextContent().trim();
            }

            if(node.getNodeName().equals(UPDATE_NODE_NAME))
            {
                update = node.getTextContent().trim();
            }

            if(node.getNodeName().equals(DELETE_NODE_NAME))
            {
                delete = node.getTextContent().trim();
            }

            if(node.getNodeName().equals(BLOB_NODE_NAME))
            {
                blob = node.getTextContent().trim();
            }

            index = new Index(name,all,add,update,delete,blob);
        }

        return index;
    }


    /**
     * 返回结果数据
     * @return
     */
    public List<Index> getTableConfigs()
    {
        return tableConfigs;
    }
}



*************************************************2.定时创建索引****************************************
如果创建索引的数据量较大,创建索引需要花很长的时间,建议创建定时任务创建索引

由于第一次是索引的全部创建,之后就可以更新索引(新增,更新,删除)即可不用每次全部创建,
所以要记录索引的最大ID和上一次更新时间

1>////////////////////////创建索引的定时任务CreateIndexJob.java:

package com.lhzq.ibms.lucene.job;

import com.htsc.abms.lucene.model.Index;
import com.htsc.abms.lucene.service.CreateIndexService;
import com.lhzq.ibms.commons.util.Configuration;
import com.lhzq.ibms.lucene.util.*;
import com.lhzq.leap.core.utils.DateUtils;
import com.lhzq.leap.core.utils.FileUtility;
import com.lhzq.leap.core.utils.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.io.File;
import java.io.IOException;
import java.math.BigDecimal;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
* 创建索引的定时任务
*/
@Service("createIndexJob")
public class CreateIndexJob
{
    /**
     * 日志
     */
   private static Logger logger = LoggerFactory.getLogger(CreateIndexJob.class);

    /**
     * 示例用户业务处理
     */
   @Autowired
   private CreateIndexService indexService;

    /**
     * 创建索引工具
     */
   private BuildIndex buildIndex;

    /**
     * 记录最大的Id和更新索引的时间
     */
   private IndexLog indexLog;

    /**
     *  全部加载索引
     */
   public String loadIndex()
   {
      StringBuffer message = new StringBuffer();
      message.append("["+DateUtils.now()+"]:开始创建索引***********!\r\n");
      logger.info("开始创建索引***************************");
      long begin=System.currentTimeMillis();
      List<Index> indexes = IndexConfigMgr.getInstance().getTableConfigs();
      try
      {
         // 先删除目录
         String indexPath = Configuration.getLuceneIndexDir();
         FileUtility.deleteDir(indexPath);
         message.append("删除index目录成功!\r\n");
         logger.info("删除index目录成功**********");

         // 创建日志文件
         CreateLog.init();
         BigDecimal maxId = null;
         for(Index index : indexes)
         {
             message.append("开始创建["+index.getName()+"]模块的索引!\r\n");
             logger.info("开始创建["+index.getName()+"]索引======");

             // 设置索引参数
             buildIndex = new BuildIndex(index.getName());
             indexLog = new IndexLog(index.getName());

             // 写入索引
             buildIndex.setDocType(BuildIndex.DOC_TYPE_CREATE);
             maxId = pageAddDoc(buildIndex,index,new HashMap());

             // 关闭
             buildIndex.close();

             // 写如参数
             after(indexLog,maxId);

             message.append("创建["+index.getName()+"]模块索引完成!\r\n");
             logger.info("创建["+index.getName()+"]索引完成=======");
         }

         message.append("["+DateUtils.now()+"]:创建索引完成***********!\r\n");
         long end=System.currentTimeMillis();
         message.append("创建索引一共花费:"+(float)(end-begin)/1000+"秒");

         logger.info("创建索引完成********************************");
      } catch (Exception e) {
         message.append("创建索引异常:"+e.getMessage());
         logger.error("加载所有的索引失败", e);
      }

      return message.toString();
   }

    /**
     * 定时更新索引
     */
   public String updateIndex()
    {
        StringBuffer message = new StringBuffer();
        long begin=System.currentTimeMillis();
        message.append("["+DateUtils.now()+"]:开始更新索引***********!\r\n");

        List<Index> indexes = IndexConfigMgr.getInstance().getTableConfigs();
        try
        {
            BigDecimal maxId = null;
            BigDecimal addMaxId = null;
            HashMap<String,Object> params = null;
            for(Index index : indexes)
            {
                message.append("开始更新["+index.getName()+"]模块的索引!\r\n");

                // 读取参数
                buildIndex = new BuildIndex(index.getName());
                indexLog = new IndexLog(index.getName());
                params = before(indexLog);

                // 拿出最大ID
                maxId =(BigDecimal)params.get("ID");

                // 添加索引
                buildIndex.setDocType(BuildIndex.DOC_TYPE_ADD);
                addMaxId = pageAddDoc(buildIndex,index,params);
                // 更新最大ID
                if(null != addMaxId){
                    maxId = addMaxId;
                }

                // 更新索引
                buildIndex.setDocType(BuildIndex.DOC_TYPE_UPDATE);
                pageAddDoc(buildIndex,index,params);

                // 删除索引
                buildIndex.setDocType(BuildIndex.DOC_TYPE_DELETE);
                pageAddDoc(buildIndex,index,params);

                // 关闭
                buildIndex.close();

                // 写如参数
                after(indexLog,maxId);
                message.append("更新["+index.getName()+"]模块索引完!\r\n");
            }

            message.append("["+DateUtils.now()+"]:更新索引完成***********!\r\n");
            long end=System.currentTimeMillis();
            message.append("更新索引花费了时间:" + (float)(end-begin)/1000+"秒");
        } catch (Exception e) {
            message.append("更新索引异常:" + e.getMessage());
            logger.error("更新索引失败", e);
        }

        return message.toString();
    }

    /**
     * 读取索引文件内容
     * @param indexLog
     * @return
     */
    private HashMap<String,Object> before(IndexLog indexLog)
    {
        HashMap<String,Object> params = new HashMap<String, Object>();
        String content  = indexLog.readText();

        if(!StringUtils.isEmpty(content))
        {
           String id = content.split(",")[0];
           String now = content.split(",")[1];

           // 封装参数
           params.put("ID", new BigDecimal(id));
           params.put("UPDATE_TIME",DateUtils.toDate(now));

           logger.info("索引库中最大的ID:"+ id+",上次更新时间:"+now);
        }

        return params;
    }

    /**
     * 写入新的最大ID和时间
     * @param indexLog
     * @param maxId
     */
    private void after(IndexLog indexLog,BigDecimal maxId)
    {
        if(null == maxId){
            return;
        }

        String now =  DateUtils.toString(new Date());
        indexLog.WriteText(maxId + "," +now );

        logger.info("写入最大的ID:"+ maxId+",记录更新时间:"+now);
    }

    /**
     * 分页操作添加索引
     * @param buildIndex
     * @param index
     * @param param
     * @return
     * @throws Exception
     */
    private BigDecimal pageAddDoc(BuildIndex buildIndex,Index index,Map param) throws IOException {
        DataPage dataPage = new DataPage(this.indexService,index.getBlob(),param);
        BigDecimal maxId = null;
        int count = 0;
        switch (buildIndex.getDocType())
        {
            case BuildIndex.DOC_TYPE_CREATE:
            {
                dataPage.setBaseSql(index.getAll());
                count = (int)Math.ceil((float)dataPage.getCount()/DataPage.PAGE_SIZE);
                dataPage.setTotalPage(count);
                for(int i =1;i<=count;i++){
                   buildIndex.addDoc(dataPage.queryPage(i));
                }
                maxId = dataPage.getMaxId();
                break;
            }
            case BuildIndex.DOC_TYPE_ADD:
            {
                dataPage.setBaseSql(index.getAdd());
                count = (int)Math.ceil((float)dataPage.getCount()/DataPage.PAGE_SIZE);
                dataPage.setTotalPage(count);
                for(int i =1;i<=count;i++){
                    buildIndex.addDoc(dataPage.queryPage(i));
                }
                maxId = dataPage.getMaxId();
                break;
            }
            case BuildIndex.DOC_TYPE_UPDATE:
            {
                dataPage.setBaseSql(index.getUpdate());
                count = (int)Math.ceil((float)dataPage.getCount()/DataPage.PAGE_SIZE);
                dataPage.setTotalPage(count);
                for(int i =1;i<=count;i++){
                    buildIndex.updateDoc(dataPage.queryPage(i));
                }
                break;
            }
            case BuildIndex.DOC_TYPE_DELETE:
            {
                dataPage.setBaseSql(index.getDelete());
                count = (int)Math.ceil((float)dataPage.getCount()/DataPage.PAGE_SIZE);
                dataPage.setTotalPage(count);
                for(int i =1;i<=count;i++){
                    buildIndex.deleteDoc(dataPage.queryPage(i));
                }
                break;
            }
        }

        return maxId;
    }

    public CreateIndexService getIndexService() {
        return indexService;
    }

    public void setIndexService(CreateIndexService indexService) {
        this.indexService = indexService;
    }
}



2>.//////////////////////////创建索引工具类BuildIndex.java:

package com.lhzq.ibms.lucene.util;

import com.lhzq.ibms.commons.util.Configuration;
import com.lhzq.leap.core.config.CommonConfig;
import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.document.Document;
import org.apache.lucene.document.Field;
import org.apache.lucene.document.TextField;
import org.apache.lucene.index.IndexWriter;
import org.apache.lucene.index.IndexWriterConfig;
import org.apache.lucene.index.Term;
import org.apache.lucene.store.Directory;
import org.apache.lucene.util.Version;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-12
* Time: 下午4:06
* Lucene创建索引工具类
*/
public class BuildIndex
{
    /**
     * 操作类型
     */
    public static final int DOC_TYPE_CREATE = 0;
    public static final int DOC_TYPE_ADD = 1;
    public static final int DOC_TYPE_UPDATE = 2;
    public static final int DOC_TYPE_DELETE = 3;

    /**
     * 索引写入器
     */
    private IndexWriter indexWriter;

    /**
     * 操作类型
     */
    private int docType;

    /**
     * 构造方法创建索引写入器
     *
     * @param name
     */
    public BuildIndex(String name) throws IOException {
        // 创建IndexWriter
        String indexPath = Configuration.getLuceneIndexDir();
        indexWriter = getIndexWriter(indexPath + "/" + name);
    }

    // 索引写入器
    private IndexWriter getIndexWriter(String indexDir) throws IOException {
        // 存储索引在硬盘中
        Directory dir = DirCenter.getDir(indexDir);

        // Version操作开始变得非常常见
        // 中文分词器的引入,好像4.7.0对庖丁等第三方分词器兼容得并不好,可能也是因为apache对原生的做了一些整合的缘故
        Analyzer analyzer = AnalyzerCenter.getAnalyzer();

        // 同时引入了IndexWriterConfig对象,封装了早期版本的一大堆参数
        IndexWriterConfig config = new IndexWriterConfig(Version.LUCENE_47, analyzer);
        IndexWriter writer = new IndexWriter(dir, config);

        return writer;
    }

    // 创建document对象
    private Document createDoc(Map<String, Object> record) throws UnsupportedEncodingException {

        Document doc = new Document();
        Iterator<String> it = record.keySet().iterator();

        String key = null;
        String value = null;
        while (it.hasNext()) {
            key = it.next();
            value = String.valueOf(record.get(key));
            doc.add(new Field(key, value, TextField.TYPE_STORED));
        }

        return doc;
    }

    // 添加索引
    public void addDoc(List<Map<String, Object>> data) throws IOException {
        for (Map<String, Object> record : data) {
            Document doc = createDoc(record);
            indexWriter.addDocument(doc);
        }
    }

    // 更新索引
    public void updateDoc(List<Map<String, Object>> data) throws IOException {
        for (Map<String, Object> record : data) {
            Document doc = createDoc(record);

            Term term = new Term("ID", "" + record.get("ID"));

            indexWriter.updateDocument(term, doc);
        }
    }

    // 删除索引
    public void deleteDoc(List<Map<String, Object>> data) throws IOException {
        for (Map<String, Object> record : data) {
            Term term = new Term("ID", "" + record.get("ID"));

            indexWriter.deleteDocuments(term);
        }
    }

    //  关闭
    public void close() throws IOException {
        if (null != this.indexWriter) {
            this.indexWriter.close();
            this.indexWriter = null;
        }
    }

    public int getDocType() {
        return docType;
    }

    public void setDocType(int docType) {
        this.docType = docType;
    }
}


3>.////////////////////////////单例拿到解析器

package com.lhzq.ibms.lucene.util;

import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.analysis.standard.StandardAnalyzer;
import org.apache.lucene.util.Version;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-15
* Time: 上午10:21
* 单例模式 获取解析器
*/
public class AnalyzerCenter
{
    private static Analyzer analyzer;

    private AnalyzerCenter(){}

    public static Analyzer getAnalyzer()
    {
         if(null == analyzer)
         {
             analyzer = new StandardAnalyzer(Version.LUCENE_47);
         }

        return  analyzer;
    }
}


4>.///////////////////////////////打开一个索引目录工具类
package com.lhzq.ibms.lucene.util;

import org.apache.lucene.store.Directory;
import org.apache.lucene.store.FSDirectory;

import java.io.File;
import java.io.IOException;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-15
* Time: 上午10:31
* 打开一个目录
*/
public class DirCenter
{
    private DirCenter(){}

    public static Directory getDir(String path) throws IOException
    {
        // 检查参数
        if(null == path)
        {
            return null;
        }

        File indexDir = new File(path);

        // 如果文件不存在,则创建目录
        if(!indexDir.exists())
        {
            indexDir.mkdir();
        }

        // 存储索引在硬盘中
        Directory dir = FSDirectory.open(indexDir);
        return dir;
    }
}


5>./////////////////////////////一次创建索引太多,会导致内存溢出,需要分页创建
package com.lhzq.ibms.lucene.util;

import com.htsc.abms.lucene.service.CreateIndexService;
import java.math.BigDecimal;
import java.util.List;
import java.util.Map;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-6-3
* Time: 下午4:09
* 包装分页查询数据
*/
public class DataPage
{
    /**
     * 每页条数
     */
    public static final int PAGE_SIZE = 20;

    /**
     * 业务操作
     */
    private CreateIndexService indexService;

    /**
     * 原始sql
     */
    private String baseSql;

    /**
     * blob字段
     */
    private String blob;

    /**
     * 参数
     */
    private Map param;

    /**
     * 最大的id
     */
    private BigDecimal maxId;

    /**
     * 总页数
     */
    private Integer totalPage;


    /**
     * 构造方法设置查询条件
     * @param indexService
     * @param blob
     * @param param
     */
    public DataPage(CreateIndexService indexService,String blob,Map param)
    {
       this.indexService = indexService;
       this.blob = blob;
       this.param = param;
    }

    /**
     * 查询一页数据
     * @param pageNo
     * @return
     */
    public List<Map<String,Object>> queryPage(int pageNo)
    {
        String sql  = "SELECT * FROM (SELECT A.*,ROWNUM RN FROM ("+this.baseSql+") A WHERE ROWNUM <= "+pageNo * PAGE_SIZE+")"
                +" WHERE RN >= "+((pageNo-1) * PAGE_SIZE + 1);

        List<Map<String,Object>> data = indexService.queryPageData(sql,this.blob,this.param);

        if(pageNo == totalPage)
        {
           this.maxId = (BigDecimal)data.get(data.size() -1).get("ID");
        }

        return  data;
    }

    /**
     * 查询总数量
     * @return
     */
    public Integer getCount()
    {
        String  sql = "SELECT COUNT(*) CNT FROM ("+this.baseSql+") A";
        return indexService.getCount(sql,this.param);
    }

    public BigDecimal getMaxId() {
        return maxId;
    }

    public void setBaseSql(String baseSql) {
        this.baseSql = baseSql;
    }

    public void setTotalPage(Integer totalPage) {
        this.totalPage = totalPage;
    }
}


6>.///////////////////////////////记录上一次更新索引的时间和最大ID,方便更新索引
package com.lhzq.ibms.lucene.util;

import com.lhzq.ibms.commons.util.Configuration;
import com.lhzq.leap.core.config.CommonConfig;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.*;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-13
* Time: 下午1:25
* 更新索引时,修改保存最大Id和更新时间
*/
public class IndexLog
{
    /**
     * 日志
     */
    private static Logger logger = LoggerFactory.getLogger(IndexLog.class);

    /**
     * 保存最大的Id和更新索引的时间的文件
     */
    private File logFile;

    /**
     * 设置日志文件
     * @param dir
     */
    public IndexLog(String dir) throws IOException
    {
        String indexPath = Configuration.getLuceneIndexDir()+"/" + dir;;
        File fileDir = new File(indexPath);

        if(!fileDir.exists()){
           fileDir.mkdir();
        }
        File file = new File(fileDir,dir + ".txt");

        if(!file.exists()){
           file.createNewFile();
        }

        logFile = file;
    }

    /**
     * 读取上一次更新索引的时间和最大ID
     * @return
     * @throws java.io.IOException
     */
    public String readText()
    {
        BufferedReader br = null;
        String content = null;
        try
        {
            br = new BufferedReader(new FileReader(logFile));
            content = br.readLine();
        }
        catch (IOException e)
        {
            logger.error("读取最大ID和上次更新索引时间失败", e);
        }
        finally
        {
            try {
                if (br != null) {
                    br.close();
                    br = null;
                }
            } catch (IOException e) {
                logger.error("读取最大ID和上次更新索引时,关闭IO失败", e);
            }
        }

        return content;
    }

    /**
     * 写入创建或者更新索引日志
     * @param text
     * @throws java.io.IOException
     */
    public void WriteText(String text)
    {
        BufferedWriter bw = null;
        try {
            bw = new BufferedWriter(new FileWriter(logFile));
            bw.write(text);
        } catch (IOException e) {
          logger.error("写入最大ID和上次更新索引失败", e);
        }
        finally
        {
           try {
             if(bw!=null){
               bw.close();
               bw = null;
             }
           } catch (IOException e) {
              logger.error("写入最大ID和上次更新索引时,关闭IO失败", e);
           }
        }
     }
}

***********************************************************3.手动创建索引******************************************
1>.////////////////////////页面:
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN">
<%@ page contentType="text/html;charset=UTF-8" language="java" %>

<%@include file="/modules/comm/loadingData.jsp"%>
<%@include file="/common/path_header.jsp" %>
<%@include file="/common/jqgrid_header.jsp" %>

<!--dwr-->
<script type="text/javascript" src="<%=path%>/dwr/engine.js"></script>
<script type="text/javascript" src="<%=path%>/dwr/util.js"></script>
<script type="text/javascript" src="<%=path%>/dwr/interface/dwrIndexManage.js"></script>

<html>

<head>
    <title>索引维护</title>
    <meta http-equiv="content-type" content="text/html; charset=UTF-8">
<style type="text/css">
    .indexInfo{
        width: 800px;
        height: 200px;
        border: 2px solid #E5E5E5;
    }
</style>
</head>
<body>

<div style="text-align: center">
        <ul  style="text-align:left;list-style-type:none;">
            <li>
                <a class="button glow button-rounded button-flat-primary button-tiny" onclick="createIndex();"> 创建索引 </a> &nbsp;&nbsp;
            </li>
            <li>
                <div class="indexInfo" id="createInfo">创建索引日志...</div>
            </li>
            <li style="margin-top: 10px;">
                <a class="button glow button-rounded button-flat-primary button-tiny" onclick="updateIndex();"> 更新索引 </a>
            </li>
            <li>
                <div class="indexInfo" id="updateInfo">更新索引日志...</div>
            </li>
        </ul>
    </div>
</body>
</html>

<script type="text/javascript">

    var interval = null;

    var time = null;

    // 创建索引
    function createIndex(){
       // 显示加载
        createDiv();

       // 创建索引
       dwrIndexManage.createIndex();

       // 延迟执行
       time = setTimeout(function(){
          interval = setInterval(showCtResult, "1000");
       },60000);
    }

    // 显示创建结果
    function showCtResult(){
        dwrIndexManage.queryCtResult(function(data){
            if(null!=data&&data!=''){
                clearTimeout(time);
                clearInterval(interval)
                setValue("createInfo",data);

                // 加载完成移出
                removeDiv();
            }
        })
    }

    // 更新索引
    function updateIndex(){
      // 显示加载
      createDiv();

      dwrIndexManage.updateIndex({
          //回调函数
          callback: function(data){
              setValue("updateInfo",data);

              // 加载完成移出
              removeDiv();
          },
          //超时,单位是毫秒,默认为20分钟,设置为0代表关闭超时
          timeout: 0,
          //超时后调用的处理函数
          errorHandler:function(message) { alert(message); }
      });
    }

    // 设置值
    function setValue(id,data){
       document.getElementById(id).innerHTML="";
       document.getElementById(id).innerHTML="<pre>"+data+"</pre>";
    }
</script>

2>.////////////////////////////////DWR操作
package com.lhzq.ibms.lucene.dwr;

import com.lhzq.ibms.lucene.job.CreateIndexJob;
import com.lhzq.ibms.lucene.util.CreateLog;
import org.springframework.beans.factory.annotation.Autowired;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-6-18
* Time: 下午5:24
* 手动索引的创建和更新
*/
public class DwrIndexManage
{
   @Autowired
   private CreateIndexJob indexJob;

    /**
     * 创建索引
     * @return
     */
   public void createIndex()
   {
      String logInfo = indexJob.loadIndex();
      CreateLog.write(logInfo);
   }

    /**
     * 查询创建索引结果
     * @return
     */
    public String queryCtResult()
    {
      return CreateLog.read().trim();
    }

    /**
     * 更新索引
     * @return
     */
    public String updateIndex()
    {
        return indexJob.updateIndex();
    }
}


3>.////////////////////////////页面返回创建操作日志:
package com.lhzq.ibms.lucene.util;

import com.lhzq.ibms.commons.util.Configuration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.*;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-13
* Time: 下午1:25
* 记录创建和更新索引的操作日志
*/
public class CreateLog
{
    /**
     * 日志
     */
    private static Logger logger = LoggerFactory.getLogger(CreateLog.class);

    /**
     * 创建日志文件
     */
    private static String path = Configuration.getLuceneIndexDir()+"/createLog.txt";

    /**
     * 创建文件
     */
    public static void init(){
        try {
            File indexDir = new File(Configuration.getLuceneIndexDir());

            // 如果文件不存在,则创建目录
            if(!indexDir.exists())
            {
                indexDir.mkdir();
            }

            // 创建文件
            File createLogFile = new File(path);
            if(!createLogFile.exists()){
                createLogFile.createNewFile();
            }
        } catch (IOException e) {
            logger.error("创建日志文件失败",e);
        }
    }

    /**
     * 读取日志文件
     * @return
     */
    public static String read()
    {
        BufferedReader br = null;
        StringBuffer log =new StringBuffer();
        try
        {
            br = new BufferedReader(new FileReader(path));
            String line = null;
            while((line = br.readLine())!=null)
            {
                log.append(line).append("\r\n");
            }
        }
        catch (IOException e)
        {
            logger.error("创建索引读取日志异常", e);
        }
        finally
        {
            try {
                if (br != null) {
                    br.close();
                }
            } catch (IOException e) {
                logger.error("创建索引读取日志,关闭IO失败", e);
            }
        }

        return log.toString();
    }

    /**
     * 写入创建或者更新索引日志
     */
    public static void write(String log)
    {
        BufferedWriter bw = null;
        try {
            bw = new BufferedWriter(new FileWriter(path));
            bw.write(log);
        } catch (IOException e) {
           logger.error("创建索引写入日志错误", e);
        }
        finally
        {
           try {
             if(bw!=null){
               bw.close();
             }
           } catch (IOException e) {
              logger.error("创建索引写入日志,关闭IO失败", e);
           }
        }
     }
}


三、查询索引
1>.///////////////////////////////////页面
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN">
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib prefix="struts" uri="/struts-tags" %>

<!--引入path java中的path和js中的path-->
<%@include file="/common/path_header.jsp" %>
<%@include file="/common/jqgrid_header.jsp" %>

<html>
<head>
    <title>全文检索</title>
    <script type="text/javascript" src="${path}/script/common/rims.js"></script>
    <meta http-equiv="content-type" content="text/html; charset=UTF-8">
</head>
<body style="text-align: center">

<!--查询条件-->
  <form id="riskReferencePoint" method="post">
    <div style="text-align: center">
     <div style="overflow:auto;zoom:1;padding:10px 0px 5px 0px;">
       <ul  style="text-align:left;list-style-type:none;">
          <li style="float:left;">
            检索信息:
            <input type="text" id="KEYWORD" name="keyword" onkeydown="if(event.keyCode==13){ enterSearch();}" size="80" />
              &nbsp;&nbsp;&nbsp;
          </li>

          <li style="float:left;">
            <a href="javascript:search();" class="button glow button-rounded button-flat-primary button-tiny" id="save"> 检索 </a>
             &nbsp;&nbsp;
          </li>
       </ul>
     </div>
    </div>
   </form>

   <!--检索规章制度结果集-->
   <table id="riskRolesGrid">

   </table>
   <br>
   <!--检索风险问题结果集-->
   <table id="riskProblemGrid">

   </table>

</body>


<script type="text/javascript">

    //
    no_data();

    // 首次加载的时候,不到后台查询数据
    function no_data()
    {
        var keyword = $("#KEYWORD").val();

        if(keyword ==undefined||keyword ==null||keyword=='')
        {
            return;
        }
    }

    //  检索规章制度
    new AbmsGrid('riskRolesGrid',{
         colNames:['id','标题','内容','文档下载'],
         colModel:[
             {
                name:'ID'
               ,key:true
               ,width:55
               ,hidden:true
             }
            ,{
                name:'NAME'
               ,width:100
             }
             ,{
                 name:'BODY'
                ,width:400
                ,formatter:function(value){
                    return "<pre>"+trimToSummary(value)+"</pre>";
                }
             }
             ,{
                 width:60
                ,align:'center'
                ,formatter:function( value,options,rowData ){
                     //自定义渲染函数
                     if(rowData.BODY==undefined||rowData.BODY==null||rowData.BODY==''){
                         return '--';
                     }
                     return '<a href="javascript:uploadRiskRules('+rowData.ID+');" style="color:#fff" class="button glow button-rounded button-flat-primary button-tiny">文档下载</a>';
                 }
             }
         ],
         postParamNames:['KEYWORD'],
         _gridDatasourceClass:'com.htsc.abms.auditrisk.web.RiskRuleDatasource',
         showPagerTool:true,
         loadDataFlag:false,
         caption:"风险规章制度"

     });

    //  检索风险问题
    new AbmsGrid('riskProblemGrid',{
        colNames:['id','风险问题', '审计意见','处罚意见','详细信息'],// ,'处罚内容'
        colModel:[
            {
                name:'ID'
               ,key:true
               ,width:55
               ,hidden:true
            },
            {
                name:'TITLE'
               ,width:220
            },
            {
                name:'CONTENTS'
               ,width:250
            },
            {
                name:'PUNISH'
               ,width:200
          }
//        ,{
//             name:'CRITERION_CONTENT'
//             ,width:200
//         }
            ,{
                width:60
                ,align:'center'
                ,formatter:function( value,options,rowData ){
                    //自定义渲染函数
                    return '<a href="javascript:findRiskProblem('+rowData.ID+');" style="color:#fff" class="button glow button-rounded button-flat-primary button-tiny">详细信息</a>';
                }
            }
        ],
        postParamNames:['KEYWORD'],
        _gridDatasourceClass:'com.htsc.abms.auditrisk.web.RiskProblemDatasource',
        showPagerTool:true,
        loadDataFlag:false,
        caption:"风险问题"
    });

    // 检索
    function search(){
        $("#riskRolesGrid").trigger("reloadGrid");
        $("#riskProblemGrid").trigger("reloadGrid");
    }

    // 显示制度详情
    function uploadRiskRules(id)
    {
        var  inputs='<input type="hidden" name="id" value="'+id+'"/>';
        jQuery('<form action="/htsc.abms/riskRules/uploadRiskRules.do" method="post">'+inputs+'</form>').appendTo('body').submit().remove();
    }

    // 显示详细信息
    function findRiskProblem(id) {
        var _url="${path}/risk/viewRiskById.do?riskId="+id;
        rims.window.showWindow(_url,900,900,null);
    }

    // 点击回车键查询
    function enterSearch(){
      $("#KEYWORD").blur();
      search();
    }

    // 截取字符串
    function trimToSummary(str){
        var endLength = 30;
        if(null==str||str==''){
           return str;
        }

        if(str.length > endLength){
            return str.substring(0,endLength) +'...';
        } else{
            return str;
        }
    }
</script>
</html>

2>./////////////////////////////////////后台数据读取
package com.htsc.abms.auditrisk.web;

import com.htsc.abms.jqgrid.model.GridData;
import com.htsc.abms.jqgrid.model.GridPostParam;
import com.htsc.abms.jqgrid.util.GridDatasourceInterface;
import com.lhzq.ibms.lucene.util.Searcher;
import com.lhzq.leap.core.utils.AppUtils;
import com.lhzq.leap.core.utils.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;

import java.util.List;
import java.util.Map;

/**
* User: 陈圣林
* Date: 14-5-27
* Time: 下午2:19
* 风险问题查选检索
*/
@Component
public class RiskProblemDatasource implements GridDatasourceInterface {
    /**
     * 日志
     */
    private static Logger logger = LoggerFactory.getLogger(RiskProblemDatasource.class);

    /**
     * 索引的字段
     */
    private static final String[] INDEX_FIELDS = {"ID", "TITLE", "CONTENTS", "PUNISH", "CRITERION_CONTENT"};

    /**
     * 根据参数查询检索信息
     *
     * @param gridPostParam
     * @return jqgrid数据对象
     */
    public GridData getGridData(GridPostParam gridPostParam) {
        // 拿到关键字参数
        String keyword = (String) gridPostParam.getParamMap().get("KEYWORD");
        if (StringUtils.isEmpty(keyword)) {
            return new GridData();
        }

        // 那到当前页
        Integer currentPage = gridPostParam.getPage();

        // 每页显示的行数
        Integer pageSize = gridPostParam.getPageSize();

        // 全文检索查询器
        Searcher searcher = null;
        List<Map<String, String>> data = null;

        // 处理关键字
        String [] keywords = AppUtils.keywords(keyword);
        try {
            searcher = new Searcher("riskProblem");
            data = searcher.search(keywords, INDEX_FIELDS);
        } catch (Exception e) {
            logger.error("全文检索异常", e);
        }

        if (AppUtils.isBlank(data)) {
            return new GridData();
        }

        // 返回当前对象
        GridData gridData = new GridData(pageSize, currentPage, data);

        return gridData;
    }
}


单个解析词,是根据单个字查询的,为了按词组查询,需要做处理

package com.lhzq.leap.core.utils;

import java.io.Serializable;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collection;
import java.util.GregorianCalendar;
import java.util.Iterator;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;

import org.apache.commons.beanutils.BeanUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
* 应用帮助工具
*/
public class AppUtils{
/**
     * 加特殊字符做整体分词
     *
     * @param keyword
     * @return
     */
    public static String[] keywords(String keyword) {
        String[] keywords = keyword.trim().split("\\s+");

        for (int i = 0; i < keywords.length; i++) {
            keywords[i] = "\"" + keywords[i] + "\"";
        }

        return keywords;
    }
}

3>.////////////////////////封装的查询器
package com.lhzq.ibms.lucene.util;

import com.lhzq.ibms.commons.util.Configuration;
import com.lhzq.leap.core.config.CommonConfig;
import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.document.Document;
import org.apache.lucene.index.DirectoryReader;
import org.apache.lucene.index.IndexReader;
import org.apache.lucene.index.IndexableField;
import org.apache.lucene.queryparser.classic.MultiFieldQueryParser;
import org.apache.lucene.search.*;
import org.apache.lucene.store.Directory;
import org.apache.lucene.util.Version;

import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
* Created with IntelliJ IDEA.
* User: 陈圣林
* Date: 14-5-12
* Time: 下午5:40
* 全文索引收索工具类
*/
public class Searcher
{
    /**
     * 最大获取的匹配文档数,比如100个总文档,
     * 你的query表达式匹配了50个,但是你传的maxCount为5,那就是选最优的前5个
     */
    private static final int MAX_COUNT = 1000;

    /**
     * 查询器
     */
    private IndexSearcher indexSearcher = null;

    /**
     * 创建索引查询器
     * @param name 索引目录
     * @throws java.io.IOException
     */
    public Searcher(String name) throws IOException
    {
         // 创建索引的位置
         String indexPath = Configuration.getLuceneIndexDir() + "/" + name;

         // 打开索引目录
         Directory indexDir = DirCenter.getDir(indexPath);

         // 读取器
         IndexReader reader = DirectoryReader.open(indexDir);

         // 创建索引
         indexSearcher = new IndexSearcher(reader);
    }


    /**
     * 根据关键字搜索
     * @param keywords 关键字
     * @return
     * @throws Exception
     */
    public List<Map<String,String>> search(String keywords,String []indexFields) throws Exception
    {
        // 解析器
        Analyzer analyzer  = AnalyzerCenter.getAnalyzer();

        MultiFieldQueryParser parser = new MultiFieldQueryParser(Version.LUCENE_47,indexFields,analyzer);

        // 查询对象
        Query query = parser.parse(keywords);

        return search(query);
    }


    /**
     * 根据多个关键字搜索
     * @param keywords 关键字
     * @return
     * @throws Exception
     */
    public List<Map<String,String>> search(String [] keywords,String []indexFields) throws Exception
    {
        // 解析器
        Analyzer analyzer  = AnalyzerCenter.getAnalyzer();

        MultiFieldQueryParser parser = new MultiFieldQueryParser(Version.LUCENE_47,indexFields,analyzer);

        // 多关键子查询
        BooleanQuery bq = new BooleanQuery();

        // 查询对象
        Query query = null;
        for(String keyword : keywords)
        {
            query=parser.parse(keyword);
            // 是表示And关系
            bq.add(query, BooleanClause.Occur.MUST);
        }

        return search(bq);
    }

    /**
     * 根据Query查询结果集
     * @param query
     * @return
     * @throws Exception
     */
    private List<Map<String,String>> search(Query query)throws Exception
    {
        // 查询匹配的前50个
        ScoreDoc[] hits = indexSearcher.search(query, null, MAX_COUNT).scoreDocs;

        // 封装检索的数据
        List<Map<String,String>>  data = new ArrayList<Map<String,String>>();
        Map<String,String> record = null;
        Document hitDoc = null;
        for (int i = 0; i < hits.length; i++) {
            hitDoc = indexSearcher.doc(hits[i].doc);
            record = getDocsItem(hitDoc);
            data.add(record);
        }

        return  data;
    }

    /**
     * 转换Doc对象为map数据结构
     * @param hitDoc  检索的doc对象
     * @return
     * @throws java.io.IOException
     */
    private Map<String,String> getDocsItem(Document hitDoc) throws IOException
    {
        // 文档的字段
        List<IndexableField> indexes = hitDoc.getFields();

        // 封装数据
        String name = null;
        String value = null;
        Map<String,String> record = new HashMap<String, String>();
        for(IndexableField  index : indexes)
        {
            name = index.name();
            value = index.stringValue();
            record.put(name,value);
        }

        return record;
    }
}




分享到:
评论

相关推荐

    全文检索Lucene 全文检索Lucene

    以上内容只是Lucene全文检索的基础知识,实际使用中还需要根据具体需求进行调整和优化。在深入学习Lucene的过程中,阅读《全文检索Lucene》这本书将是十分有益的,它将帮助你更好地理解和掌握Lucene的核心概念及其...

    [HeyJava][传智播客]全文检索Lucene源码

    【标题】:“HeyJava传智播客全文检索Lucene源码” 【描述】:这个资源主要聚焦于Lucene,一个广泛使用的全文检索库,由Apache软件基金会开发并维护。通过学习这部分内容,你可以深入理解Lucene如何实现高效的文本...

    全文检索 lucene 3.0

    总的来说,全文检索Lucene 3.0是一个强大而灵活的工具,它简化了文本搜索的复杂性,提高了搜索效率,为各种应用程序提供了高效的全文检索解决方案。通过深入学习和实践,开发者可以利用Lucene构建出满足用户需求的...

    最新全文检索 lucene-5.2.1 入门经典实例

    《最新全文检索 Lucene-5.2.1 入门经典实例》 Lucene是一个开源的全文检索库,由Apache软件基金会开发,广泛应用于各种信息检索系统。在5.2.1版本中,Lucene提供了更为高效和强大的搜索功能,为开发者提供了构建...

    全文检索 lucene-5.2.1 入门Eclipse工程实例

    Eclipse工程文件,全文检索 lucene-5.2.1 入门Eclipse工程实例,福利放送,与lucene3结果比对

    全文检索Lucene

    **全文检索Lucene** Lucene是Apache软件基金会的开源项目之一,它是一个强大的、高性能的全文检索库。作为Java编写的基础组件,Lucene为开发者提供了实现全文搜索功能所需的底层算法和数据结构。这个库不仅实现了...

    SpringMvc+Lucene全文检索

    **Spring MVC + Lucene 全文检索** 在现代Web应用中,实现高效的全文检索功能是提升用户体验的关键之一。本文将详细介绍如何使用Spring MVC框架结合Apache Lucene库来构建一个强大的全文检索系统。首先,让我们了解...

    全文检索系统(Lucene)

    **全文检索系统与Lucene** 全文检索系统是一种用于在大量文本数据中快速查找相关信息的工具。它通过索引文本中的关键词来实现高效的搜索性能,使得用户可以输入任意词汇或短语,系统能在短时间内返回最相关的文档。...

    全文检索 lucene

    Lucene从问世之后,引发了开放源代码社群的巨大反响,程序员们不仅使用它构建具体的全文检索应用,而且将之集成到各种系统软件中去,以及构建Web应用,甚至某些商业软件也采用了Lucene作为其内部全文检索子系统的...

    中文分词及其在基于Lucene的全文检索中的应用

    《中文分词及其在基于Lucene的全文检索中的应用》这篇论文主要探讨了中文分词在全文检索系统,特别是基于Lucene平台的应用。全文检索技术是现代信息检索领域的重要组成部分,而Lucene作为一款开源的全文检索引擎框架...

    使用zend Framework的lucene进行全文检索

    在本文中,我们将探讨如何使用Zend Framework的Lucene模块进行全文检索,特别是针对中文分词的处理。全文检索是提高网站或应用搜索功能的关键技术,它允许用户输入任意词汇,系统能够快速找到与之相关的内容。Zend ...

    基于Lucene和IK Analyzer的全文检索框架搭建

    介绍了全文检索的基本概念,详细说明了全文检索框架Lucene的用法,并提供了样例代码。

    Lucene全文检索引擎

    **Lucene全文检索引擎** Lucene是Apache软件基金会的一个开源项目,它是一个高性能、全文本搜索引擎库,可以被集成到各种应用中实现全文检索功能。Lucene提供了完整的搜索功能实现,包括索引创建、文档存储、查询...

    Lucene.Net 实现全文检索

    在本案例中,我们将在 .Net MVC4 框架上使用 Lucene.Net 来构建一个全文检索系统。 首先,我们需要理解全文检索的基本概念。全文检索是指在文档集合中,根据用户输入的查询词,查找包含这些词的文档。与传统的...

    Spring+SpringMVC+Mybatis+全文检索lucene+shiro实现的博客系统

    使用 SSM(Spring、SpringMVC、MyBatis)框架,MVC 三层结构、Lucene全文检索引擎、Junit 4单元测试、logback日志框架、Druid数据库连接池、Shiro安全框架的一个博文系统;

    开放源代码的全文检索引擎Lucene

    开放源代码的全文检索引擎Lucene开放源代码的全文检索引擎Lucene开放源代码的全文检索引擎Lucene

Global site tag (gtag.js) - Google Analytics