使用Lucene.NET实现简单的站内搜索

  1. 导入Lucene.NET 开发包

    Lucene 是apache软件基金会一个开放源代码的全文检索引擎工具包,是一个全文检索引擎的架构,提供了完整的查询引擎和索引引擎,部分文本分析引擎。Lucene的目的是为软件开发人员提供一个简单易用的工具包,以方便的在目标系统中实现全文检索的功能,或者是以此为基础建立起完整的全文检索引擎。Lucene.Net 是 .NET 版的Lucene。

    你可以在这里下载到最新的Lucene.NET

  2. 创建索引、更新索引、删除索引

  3. 搜索,根据索引查找

     using System;
    using Lucene.Net.Store;
    using Lucene.Net.Index;
    using Lucene.Net.Analysis.PanGu;
    using Lucene.Net.Documents; namespace BLL
    {
    class IndexHelper
    {
    /// <summary>
    /// 日志小助手
    /// </summary>
    static Common.LogHelper logger = new Common.LogHelper(typeof(SearchBLL));
    /// <summary>
    /// 索引保存的位置,保存在配置文件中从配置文件读取
    /// </summary>
    static string indexPath = Common.ConfigurationHelper.AppSettingMapPath("IndexPath"); /// <summary>
    /// 创建索引文件或更新索引文件
    /// </summary>
    /// <param name="item">索引信息</param>
    public static void CreateIndex(Model.HelperModel.IndexFileHelper item)
    {
    try
    {
    //索引存储库
    FSDirectory directory = FSDirectory.Open(new System.IO.DirectoryInfo(indexPath), new NativeFSLockFactory());
    //判断索引是否存在
    bool isUpdate = IndexReader.IndexExists(directory);
    if (isUpdate)
    {
    //如果索引目录被锁定(比如索引过程中程序异常退出),则首先解锁
    if (IndexWriter.IsLocked(directory))
    {
    //解锁索引库
    IndexWriter.Unlock(directory);
    }
    }
    //创建IndexWriter对象,添加索引
    IndexWriter writer = new IndexWriter(directory, new PanGuAnalyzer(), !isUpdate, Lucene.Net.Index.IndexWriter.MaxFieldLength.UNLIMITED);
    //获取新闻 title部分
    string title = item.FileTitle;
    //获取新闻主内容
    string body = item.FileContent;
    //为避免重复索引,所以先删除number=i的记录,再重新添加
    //尤其是更新的话,更是必须要先删除之前的索引
    writer.DeleteDocuments(new Term("id", item.FileName));
    //创建索引文件 Document
    Document document = new Document();
    //只有对需要全文检索的字段才ANALYZED
    //添加id字段
    document.Add(new Field("id", item.FileName, Field.Store.YES, Field.Index.NOT_ANALYZED));
    //添加title字段
    document.Add(new Field("title", title, Field.Store.YES, Field.Index.NOT_ANALYZED));
    //添加body字段
    document.Add(new Field("body", body, Field.Store.YES, Field.Index.ANALYZED, Lucene.Net.Documents.Field.TermVector.WITH_POSITIONS_OFFSETS));
    //添加url字段
    document.Add(new Field("url", item.FilePath, Field.Store.YES, Field.Index.NOT_ANALYZED));
    //写入索引库
    writer.AddDocument(document);
    //关闭资源
    writer.Close();
    //不要忘了Close,否则索引结果搜不到
    directory.Close();
    //记录日志
    logger.Debug(String.Format("索引{0}创建成功",item.FileName));
    }
    catch (SystemException ex)
    {
    //记录错误日志
    logger.Error(ex);
    throw;
    }
    catch (Exception ex)
    {
    //记录错误日志
    logger.Error(ex);
    throw;
    }
    } /// <summary>
    /// 根据id删除相应索引
    /// </summary>
    /// <param name="guid">要删除的索引id</param>
    public static void DeleteIndex(string guid)
    {
    try
    {
    ////索引存储库
    FSDirectory directory = FSDirectory.Open(new System.IO.DirectoryInfo(indexPath), new NativeFSLockFactory());
    //判断索引库是否存在索引
    bool isUpdate = IndexReader.IndexExists(directory);
    if (isUpdate)
    {
    //如果索引目录被锁定(比如索引过程中程序异常退出),则首先解锁
    if (IndexWriter.IsLocked(directory))
    {
    IndexWriter.Unlock(directory);
    }
    }
    IndexWriter writer = new IndexWriter(directory, new PanGuAnalyzer(), !isUpdate, Lucene.Net.Index.IndexWriter.MaxFieldLength.UNLIMITED);
    //删除索引文件
    writer.DeleteDocuments(new Term("id", guid));
    writer.Close();
    directory.Close();//不要忘了Close,否则索引结果搜不到
    logger.Debug(String.Format("删除索引{0}成功", guid));
    }
    catch (Exception ex)
    {
    //记录日志
    logger.Error(ex);
    //抛出异常
    throw;
    }
    }
    }
    }

    IndexHelper 添加、更新、删除索引

     using Lucene.Net.Analysis;
    using Lucene.Net.Analysis.PanGu;
    using Lucene.Net.Documents;
    using Lucene.Net.Index;
    using Lucene.Net.Search;
    using Lucene.Net.Store;
    using Model.HelperModel;
    using System;
    using System.Collections.Generic; namespace BLL
    {
    public static class SearchBLL
    {
    //一个类中可能会有多处输出到日志,多处需要记录日志,常将logger做成static 静态变量
    /// <summary>
    /// 日志助手
    /// </summary>
    static Common.LogHelper logger = new Common.LogHelper(typeof(SearchBLL));
    /// <summary>
    /// 索引保存位置
    /// </summary>
    static string indexPath = Common.ConfigurationHelper.AppSettingMapPath("IndexPath");
    /// <summary>
    /// 搜索
    /// </summary>
    /// <param name="keywords">用户搜索的关键词</param>
    /// <returns>返回搜索的结果</returns>
    public static List<SearchResult> Search(string keywords)
    {
    try
    {
    //索引存储库
    FSDirectory directory = FSDirectory.Open(new System.IO.DirectoryInfo(indexPath), new NoLockFactory());
    //创建IndexReader对象
    IndexReader reader = IndexReader.Open(directory, true);
    //创建IndexSearcher对象
    IndexSearcher searcher = new IndexSearcher(reader);
    //新建PhraseQuery 查询对象
    PhraseQuery query = new PhraseQuery();
    //把用户输入的关键词进行拆词
    foreach (string word in SplitWord(keywords))
    {
    //添加搜索关键词
    query.Add(new Term("body", word));
    }
    //设置分词间距为100字之内
    query.SetSlop();
    TopScoreDocCollector collector = TopScoreDocCollector.create(, true);
    //根据查询条件查询结果
    searcher.Search(query, null, collector);
    //搜索到的ScoreDoc结果
    ScoreDoc[] docs = collector.TopDocs(, collector.GetTotalHits()).scoreDocs;
    //保存搜索结果的list
    List<SearchResult> listResult = new List<SearchResult>();
    for (int i = ; i < docs.Length; i++)
    {
    //取到文档的编号(主键,这个是Lucene .net分配的)
    //检索结果中只有文档的id,如果要取Document,则需要Doc再去取
    //降低内容占用
    int docId = docs[i].doc;
    //根据id找Document
    Document doc = searcher.Doc(docId);
    string number = doc.Get("id");
    string title = doc.Get("title");
    string body = doc.Get("body");
    string url = doc.Get("url");
    //建立一个搜索结果对象
    SearchResult result = new SearchResult();
    result.Number = number;
    result.Title = title;
    result.BodyPreview = Preview(body, keywords);
    result.Url = url;
    //添加到结果列表
    listResult.Add(result);
    }
    if (listResult.Count == )
    {
    return null;
    }
    else
    {
    return listResult;
    }
    }
    catch (SystemException ex)
    {
    logger.Error(ex);
    return null;
    }
    catch (Exception ex)
    {
    logger.Error(ex);
    return null;
    }
    } /// <summary>
    /// 获取内容预览
    /// </summary>
    /// <param name="body">内容</param>
    /// <param name="keyword">关键词</param>
    /// <returns></returns>
    private static string Preview(string body, string keyword)
    {
    //创建HTMLFormatter,参数为高亮单词的前后缀
    PanGu.HighLight.SimpleHTMLFormatter simpleHTMLFormatter = new PanGu.HighLight.SimpleHTMLFormatter("<font color=\"red\">", "</font>");
    //创建 Highlighter ,输入HTMLFormatter 和 盘古分词对象Semgent
    PanGu.HighLight.Highlighter highlighter = new PanGu.HighLight.Highlighter(simpleHTMLFormatter, new PanGu.Segment());
    //设置每个摘要段的字符数
    highlighter.FragmentSize = ;
    //获取最匹配的摘要段
    string bodyPreview = highlighter.GetBestFragment(keyword, body);
    return bodyPreview;
    } /// <summary>
    /// 盘古分词,对用户输入的搜索关键词进行分词
    /// </summary>
    /// <param name="str">用户输入的关键词</param>
    /// <returns>分词之后的结果组成的数组</returns>
    private static string[] SplitWord(string str)
    {
    List<string> list = new List<string>();
    Analyzer analyzer = new PanGuAnalyzer();
    TokenStream tokenStream = analyzer.TokenStream("", new System.IO.StringReader(str));
    Lucene.Net.Analysis.Token token = null;
    while ((token = tokenStream.Next()) != null)
    {
    list.Add(token.TermText());
    }
    return list.ToArray();
    }
    }
    }

    Search 通过查找索引实现搜索

     namespace Model.HelperModel
    {
    public class SearchResult
    {
    public string Number { get; set; } public string Title { get; set; } public string BodyPreview { get; set; } public string Url { get; set; }
    }
    }

    SearchResult 模型

使用Lucene.NET实现简单的站内搜索的更多相关文章

  1. es简单打造站内搜索

    最近挺忙的,在外出差,又同时干两个项目.白天一个晚上一个,特别是白天做的项目,马上就要上线了,在客户这里 三天两头开会,问题很多真的很想好好静下来怼代码,半夜做梦都能fix bugs~ 和客户交流真的 ...

  2. 基于lucene.net 和ICTCLAS2014的站内搜索的实现1

    Lucene.net是一个搜索引擎的框架,它自身并不能实现搜索.须要我们自己在当中实现索引的建立,索引的查找.全部这些都是依据它自身提供的API来实现.Lucene.net本身是基于java的,可是经 ...

  3. Lucene.net站内搜索—4、搜索引擎第一版技术储备(简单介绍Log4Net、生产者消费者模式)

    目录 Lucene.net站内搜索—1.SEO优化 Lucene.net站内搜索—2.Lucene.Net简介和分词Lucene.net站内搜索—3.最简单搜索引擎代码Lucene.net站内搜索—4 ...

  4. Lucene.net站内搜索—3、最简单搜索引擎代码

    目录 Lucene.net站内搜索—1.SEO优化 Lucene.net站内搜索—2.Lucene.Net简介和分词Lucene.net站内搜索—3.最简单搜索引擎代码Lucene.net站内搜索—4 ...

  5. Lucene.net站内搜索—6、站内搜索第二版

    目录 Lucene.net站内搜索—1.SEO优化 Lucene.net站内搜索—2.Lucene.Net简介和分词Lucene.net站内搜索—3.最简单搜索引擎代码Lucene.net站内搜索—4 ...

  6. Lucene.net站内搜索—5、搜索引擎第一版实现

    目录 Lucene.net站内搜索—1.SEO优化 Lucene.net站内搜索—2.Lucene.Net简介和分词Lucene.net站内搜索—3.最简单搜索引擎代码Lucene.net站内搜索—4 ...

  7. Lucene.net站内搜索—2、Lucene.Net简介和分词

    目录 Lucene.net站内搜索—1.SEO优化 Lucene.net站内搜索—2.Lucene.Net简介和分词Lucene.net站内搜索—3.最简单搜索引擎代码Lucene.net站内搜索—4 ...

  8. Lucene.net站内搜索—1、SEO优化

    目录 Lucene.net站内搜索—1.SEO优化 Lucene.net站内搜索—2.Lucene.Net简介和分词Lucene.net站内搜索—3.最简单搜索引擎代码Lucene.net站内搜索—4 ...

  9. 站内搜索——Lucene +盘古分词

    为了方便的学习站内搜索,下面我来演示一个MVC项目. 1.首先在项目中[添加引入]三个程序集和[Dict]文件夹,并新建一个[分词内容存放目录] Lucene.Net.dll.PanGu.dll.Pa ...

随机推荐

  1. Codeforces Round #237 (Div. 2) C. Restore Graph(水构造)

    题目大意 一个含有 n 个顶点的无向图,顶点编号为 1~n.给出一个距离数组:d[i] 表示顶点 i 距离图中某个定点的最短距离.这个图有个限制:每个点的度不能超过 k 现在,请构造一个这样的无向图, ...

  2. Node.js 的module 系统

    相较于原生的JavaScript,不同的JavaScript文件之间很难共享变量.有鉴于此,Node.js在JavaScript的基础上进行了扩充,引入了require,exports,module三 ...

  3. javascript文件夹选择框的两种解决方案

    javascript文件夹选择框的两种解决方案 解决方案1:调用windows 的shell,但会有安全问题. * browseFolder.js * 该文件定义了BrowseFolder()函数,它 ...

  4. linux2.6.24内核源代码分析(2)——扒一扒网络数据包在链路层的流向路径之一

    在2.6.24内核中链路层接收网络数据包出现了两种方法,第一种是传统方法,利用中断来接收网络数据包,适用于低速设备:第二种是New Api(简称NAPI)方法,利用了中断+轮询的方法来接收网络数据包, ...

  5. SVN上传代码

    概述 SVN上传代码 使用SVN工具上传代码到新浪SAE 1.下载SVN工具http://tortoisesvn.net/downloads.html 2.创建一个文件夹,Checkout,输入SAE ...

  6. DDD:Can I DDD?

    下面是<实现领域驱动>的作者给出的一段话: You can implement DDD if you have: A passion for creating excellent soft ...

  7. Hadoop第13周练习—HBase作业

    1    :举例子说明HBase相对简单 1.1 1.2     回答 2    :设计HBase存储站内短信 2.1 2.2     回答 书面作业1:举例子说明HBase相对简单 请举出一例子,使 ...

  8. 利用PS自动切图、支持svg且支持icoMoon——再也不用四处去转格式了

    今天想导出svg格式的图片支持webFont,结果AI打不开了,文件好像损坏了,于是就想办法在PS里面导出. 网上搜索到一篇文章,腾讯的 http://isux.tencent.com/ps-phot ...

  9. SQL Server中的事务日志管理(5/9):完整恢复模式里的日志管理

    当一切正常时,没有必要特别留意什么是事务日志,它是如何工作的.你只要确保每个数据库都有正确的备份.当出现问题时,事务日志的理解对于采取修正操作是重要的,尤其在需要紧急恢复数据库到指定点时.这系列文章会 ...

  10. 自定义JSTL标签和函数库

    一.自定义JSTL标签 1.编写标签处理类: (1)实现 SimpleTag 接口,通过 setJspContext()方法可以获取到 jspContext 对象,实际上也是 pageContext ...