再有诸如搜索结果分页

目录:ASP.NET MVC集团级实战目录

像www.verycd.com、微博、Taobao、京东都有落到实处站内寻觅功效,站内寻觅无论在质量和客户体验上都万分不利,本节,通过行使Lucene.Net来促成站内搜索。

亲自去做效果预览如下图10-22~10-24所示。

图片 1

图10-22

 图片 2

图10-23

 图片 3

图10-24

在10.4节,已经做到了索求的第叁个本子,可是还可能有好多地方须求优化。比方说,作者要计算首要词寻找的效能高的词,也即热词,以至像百度查寻那样,在输加入关贸总协定社团键字后,会自动把寻觅相关的热词自动以下拉列表的样式带出去。还应该有诸如找寻结果分页,查看小表明细等。

10.5.1 热词总结

思路:

1、 
首先,大家脑英里要简美素佳儿点:搜索关键字的总结,实时性是不高的。也便是说我们能够定时的去开展总括。

2、  顾客的每二遍寻觅记录,大家都须要存起来,那样技术够计算得到。

从第1点,大家脑海中就展会现一张汇总总计表,从第2点中,大家会想到利用一张搜索记录明细表。那方案就很明知道,只必要准时的从明细表中Group
by查询,然后把询问结构放到汇总表中。怎么放到汇总表中?是直接Update更新吗?其实大家得以有越来越高速的方式,那正是对集中表先举行truncate,然后再扩充insert操作。

表10-1 搜索汇总总括表SearchTotals

字段名称

字段类型

说明

Id

char(36)

主键,采用Guid方式存储

KeyWords

nvarchar(50)

搜索关键字

SearchCounts

int

搜索次数

表10-2 寻找明细表SearchDetails

字段名称

字段类型

说明

Id

char(36)

主键,采用Guid方式存储

KeyWords

nvarchar(50)

搜索关键字

SearchDateTime

datetime

搜索时间

操作步骤:

(1卡塔 尔(英语:State of Qatar)在Models文件夹中,新建七个类SearchTotal、SearchDetail。

SearchTotal.cs代码:

using System;
using System.ComponentModel.DataAnnotations;

namespace SearchDemo.Models
{
    public class SearchTotal
    {
        public Guid Id { get; set; }
        [StringLength(50)]
        public string KeyWords { get; set; }
        public int SearchCounts { get; set; }
    }
}

SearchDetail.cs代码:

using System;
using System.ComponentModel.DataAnnotations;

namespace SearchDemo.Models
{
    public class SearchDetail
    {
        public Guid Id { get; set; }
        [StringLength(50)]
        public string KeyWords { get; set; }
        public Nullable<DateTime> SearchDateTime { get; set; }
    }
}

(2卡塔 尔(英语:State of Qatar)改正Search德姆oContext类,新增了质量SearchTotal、SearchDetail。

using System.Data.Entity;

namespace SearchDemo.Models
{
    public class SearchDemoContext : DbContext
    {
        public SearchDemoContext() : base("name=SearchDemoContext") { }
        public DbSet<Article> Article { get; set; }
        //下面两个属性是新增加的
        public DbSet<SearchTotal> SearchTotal { get; set; }
        public DbSet<SearchDetail> SearchDetail { get; set; }
    }
}

3卡塔尔国更新数据库

是因为更正了EF上下文,新扩大了多个模型类,所以必要张开搬迁更新数据库操作。

将应用程序重新编写翻译,然后采用工具->库程序包微处理机->程序包管控台。

开拓调控台,输入enable-migrations -force
,然后回车。回车的前边会在类型种类财富微处理机中会现身Migrations文件夹,张开Configuration.cs
文件,将AutomaticMigrationsEnabled 值改为 true,然后在调节台南输入
update-database
运转。操作完毕之后,会在数据库Search德姆o中多新建两张表SearchTotals、SearchDetails,而本来的Articles表保持不改变。如图10-20所示。

 图片 4

图10-20

(4卡塔 尔(英语:State of Qatar)保存找出记录

顾客在历次搜寻的时候,要把找寻记录存入SearchDetails表中。为了便利,这里小编是在客商每趟点击寻觅之后就立即往SearchDetails表中插入记录了,也正是同步操作,而实际上,如若为了进步查找的频率,大家能够运用异步操作,即把搜索记录的数量先写入redis队列中,后台再开荒叁个线程来监听redis队列,然后把队列中的搜索记录数据写入到数码表中。因为在每一次点击搜索的时候,大家把记录往redis写和把记录第一手往关系型数据库中写的频率是离开相当大的。

 //先将搜索的词插入到明细表。
            SearchDetail _SearchDetail = new SearchDetail { Id = Guid.NewGuid(), KeyWords = kw, SearchDateTime = DateTime.Now };
            db.SearchDetail.Add(_SearchDetail);
            int r = db.SaveChanges();

(5卡塔 尔(英语:State of Qatar)按期更新SearchTotals表记录

看来这种依期任务操作,这里可以行使Quartz.Net框架,为了便于,我把Quartz.Net的Job寄宿在调节台程序中,而事实上中国人民解放军海军事工业程高校业作中,笔者则更赞成于将其寄宿在Windows服务中。就算有不可能缺乏,能够把这些依期更新SearchTotals表记录的程序安顿到独门的服务器,那样能够缓慢解决Web服务器的压力。

  1. 新建调整台程序QuartzNet,加多Quartz.dll和Common.Logging.dll的顺序集援用,这里运用Database
    First的情势,增添ADO.NET实体数据模型,把表SearchTotals、SearchDetails增多进来。

2.增多KeyWordsTotalService.cs类,里面封装五个方法,清空SearchTotals表,然后把SearchDetails表的分组查询结构插入到SearchTotals表,这里作者只计算近30天内的找出明细。

namespace QuartzNet
{
    public class KeyWordsTotalService
    {
        private SearchDemoEntities db = new SearchDemoEntities();
        /// <summary>
        /// 将统计的明细表的数据插入。
        /// </summary>
        /// <returns></returns>
        public bool InsertKeyWordsRank()
        {
            string sql = "insert into SearchTotals(Id,KeyWords,SearchCounts) select newid(),KeyWords,count(*)  from SearchDetails where DateDiff(day,SearchDetails.SearchDateTime,
getdate())<=30 group by SearchDetails.KeyWords";
            return this.db.Database.ExecuteSqlCommand(sql) > 0;
        }
        /// <summary>
        /// 删除汇总中的数据。
        /// </summary>
        /// <returns></returns>
        public bool DeleteAllKeyWordsRank()
        {
            string sql = "truncate table SearchTotals";
            return this.db.Database.ExecuteSqlCommand(sql) > 0;
        }
    }
}
  1. 添加TotalJob.cs类,继承Ijob接口,并实现Execute方法。

    namespace QuartzNet
    {

     public class TotalJob : IJob
     {
         /// <summary>
         /// 将明细表中的数据插入到汇总表中。
         /// </summary>
         /// <param name="context"></param>
         public void Execute(JobExecutionContext context)
         {
             KeyWordsTotalService bll = new KeyWordsTotalService();
             bll.DeleteAllKeyWordsRank();
             bll.InsertKeyWordsRank();
         }
     }
    

    }

4.修改Program.cs类

using Quartz;
using Quartz.Impl;
using System;

namespace QuartzNet
{
    class Program
    {
        static void Main(string[] args)
        {
            IScheduler sched;
            ISchedulerFactory sf = new StdSchedulerFactory();
            sched = sf.GetScheduler();
            JobDetail job = new JobDetail("job1", "group1", typeof(TotalJob));//IndexJob为实现了IJob接口的类
            DateTime ts = TriggerUtils.GetNextGivenSecondDate(null, 5);//5秒后开始第一次运行
            TimeSpan interval = TimeSpan.FromSeconds(50);//每隔50秒执行一次
            Trigger trigger = new SimpleTrigger("trigger1", "group1", "job1", "group1", ts, null,
                                                    SimpleTrigger.RepeatIndefinitely, interval);//每若干时间运行一次,时间间隔可以放到配置文件中指定

            sched.AddJob(job, true);
            sched.ScheduleJob(trigger);
            sched.Start();
            Console.ReadKey();
        }
    }
}

那边本身是一贯把Job和安插都一向写到代码中了,理由依旧因为平价。而其实工作中,大家应有把那个音信尽量写到配置文件中,那样前边退换起来方便,无需改革代码,只必要改正配置文件。

为了尽早见效,笔者这里是每间距50秒就展开了壹回计算操作,而在其实使用中,大家的日子间距大概是多少个钟头以致一天,因为像这么的大额总结,对实时性的必要不高,我们得以尽量收缩对数据库的IO读写次数。

保险运营调节台程序QuartzNet,然后我们去开展搜寻操作,那样后台就准期的变迁了查找总结记录。

10.5.2 热搜

10.5.2.1 呈现热门排名

实则正是从表SearchTotals中依照搜索次数进行降序排列,然后收取数条记下而已。

LastSearch调整器中的Index方法中增加如下代码:

var keyWords = db.SearchTotal.OrderByDescending(a => a.SearchCounts).Select(x => x.KeyWords).Skip(0).Take(6).ToList();
            ViewBag.KeyWords = keyWords;

View视图中

<div id="divKeyWords">热门搜索:@if (ViewBag.KeyWords != null) {
             foreach (string v in ViewBag.KeyWords) { 
              <a href="#">@v</a>
             }
         }</div>

接下去,小编想要达成如下图10-21所示的成效:

图片 5 

图10-21

当自身点击三个热词的时候,自动加载到文本框,并点击“搜索”按键。

在View中增加代码:

<script type="text/javascript">
    $(function () {
        $("#divKeyWords a").click(function () {
            $("#txtSearch").val($(this).html());
            $("#btnSearch").click();
        });
});
</script>
10.5.2.2 搜索下拉框

那边小编引进四个第三方js框架Autocomplete,它能在文本框中输入文字的时候,自动从后台抓去数据下拉列表。

云盘中自己提供了Autocomplete.rar,将其解压,然后拷贝到Search德姆o项目中的lib目录下。

在Search德姆o项目中的KeyWordsTotalService.cs类中加多方法

using System;
using System.Collections.Generic;
using System.Data.SqlClient;
using System.Linq;

namespace SearchDemo.Common
{
    public class KeyWordsTotalService
    {
        private SearchDemoContext db = new SearchDemoContext();

        public List<string> GetSearchMsg(string term)
        {
            try
            {
                //存在SQL注入的安全隐患
                //string sql = "select KeyWords from SearchTotals where KeyWords like '"+term.Trim()+"%'";
                //return db.Database.SqlQuery<string>(sql).ToList();
                string sql = "select KeyWords from SearchTotals where KeyWords like @term";
                return db.Database.SqlQuery<string>(sql, new SqlParameter("@term", term+"%")).ToList();
            }
            catch (Exception ex)
            {
                throw new Exception(ex.Message);
            }
        }
    }
}

接下来在LastSearch调节器中增添方法:

     /// <summary>
        /// 获取客户列表 模糊查询
        /// </summary>
        /// <param name="term"></param>
        /// <returns></returns>
        public string GetKeyWordsList(string term)
        {
            if (string.IsNullOrWhiteSpace(term))
                return null;

            var list = new KeyWordsTotalService().GetSearchMsg(term);
            //序列化对象
            //尽量不要用JavaScriptSerializer,为什么?性能差,完全可用Newtonsoft.Json来代替
            //System.Web.Script.Serialization.JavaScriptSerializer js = new System.Web.Script.Serialization.JavaScriptSerializer();
            //return js.Serialize(list.ToArray());
            return JsonConvert.SerializeObject(list.ToArray());
        }

咱俩来看View:

<link href="~/lib/Autocomplete/css/ui-lightness/jquery-ui-1.8.17.custom.css" rel="stylesheet" />
<script src="~/lib/Autocomplete/js/jquery-ui-1.8.17.custom.min.js"></script>
<script type="text/javascript">
    $(function () {
        $("#divKeyWords a").click(function () {
            $("#txtSearch").val($(this).html());
            $("#btnSearch").click();
        });
        getKeyWordsList("txtSearch");
    });
    //自动加载搜索列表
    function getKeyWordsList(txt) {
        if (txt == undefined || txt == "")
            return;
        $("#" + txt).autocomplete({
            source: "/LastSearch/GetKeyWordsList",
            minLength: 1
        });
    }
</script>

10.5.3 标题和内容都扶持搜索并高亮体现

在10.4中,只援救在内容中对主要词举办检索,而实际上,我们可能既要援助在标题中搜索,也要在内容中搜寻。

此间引进了BooleanQuery,大家的查询条件也增多了二个titleQuery。

找出方法中,如下代码有改进:

 PhraseQuery query = new PhraseQuery();//查询条件
            PhraseQuery titleQuery = new PhraseQuery();//标题查询条件
            List<string> lstkw = LuceneHelper.PanGuSplitWord(kw);//对用户输入的搜索条件进行拆分。

            foreach (string word in lstkw)            {
                query.Add(new Term("Content", word));//contains("Content",word)
                titleQuery.Add(new Term("Title", word));
            }
            query.SetSlop(100);//两个词的距离大于100(经验值)就不放入搜索结果,因为距离太远相关度就不高了

            BooleanQuery bq = new BooleanQuery();
            //Occur.Should 表示 Or , Must 表示 and 运算
            bq.Add(query, BooleanClause.Occur.SHOULD);
            bq.Add(titleQuery, BooleanClause.Occur.SHOULD);

            TopScoreDocCollector collector = TopScoreDocCollector.create(1000, true);//盛放查询结果的容器
            searcher.Search(bq, null, collector);//使用query这个查询条件进行搜索,搜索结果放入collector

10.5.4 与查询、或查询、分页

前方我们在查找的时候,其实选拔的都以与查询,也便是说,作者输入“诸葛武侯周公瑾”,则只会招来出,既存在诸葛武侯,又存在周公瑾的记录。那么一时候,大家是想询问存在诸葛武侯也许周郎的笔录的,那也正是所谓的或询问。

自身在分界面增添一个复选框“或询问”,来让客户决定运用何种方法举办询问。

关于分页,这里运用MvcPager,关于MvcPager的选择办法请参见4.6.3。

View完整代码预览:

图片 6图片 7

@{
    ViewBag.Title = "Index";
}
@model PagedList<SearchDemo.Models.SearchResult>
@using Webdiyer.WebControls.Mvc;
@using SearchDemo.Models;
<style type="text/css">
.search-text2{ display:block; width:528px; height:26px; line-height:26px; float:left; margin:3px 5px; border:1px solid gray; outline:none; font-family:'Microsoft Yahei'; font-size:14px;}
.search-btn2{width:102px; height:32px; line-height:32px; cursor:pointer; border:0px; background-color:#d6000f;font-family:'Microsoft Yahei'; font-size:16px;color:#f3f3f3;}
.search-list-con{width:640px; background-color:#fff; overflow:hidden; margin-top:0px; padding-bottom:15px; padding-top:5px;}
.search-list{width:600px; overflow:hidden; margin:15px 20px 0px 20px;}
.search-list dt{font-family:'Microsoft Yahei'; font-size:16px; line-height:20px; margin-bottom:7px; font-weight:normal;}
.search-list dt a{color:#2981a9;}
.search-list dt a em{ font-style:normal; color:#cc0000;}
#divKeyWords {text-align:left;width:520px;padding-left:4px;}
#divKeyWords a {text-decoration:none;}
#divKeyWords a:hover {color:red;}
</style>
<link href="~/lib/Autocomplete/css/ui-lightness/jquery-ui-1.8.17.custom.css" rel="stylesheet" />
@using(@Html.BeginForm(null, null, FormMethod.Get))
{
    @Html.Hidden("hidfIsOr")
    <div>@Html.TextBox("txtSearch", null, new { @class="search-text2"})<input type="submit" value="搜索" name="btnSearch" id="btnSearch"  class="search-btn2"/><input type="checkbox" id="isOr" value="false"/>或查询</div>
    <div id="divKeyWords">热门搜索:@if (ViewBag.KeyWords != null) {
             foreach (string v in ViewBag.KeyWords) { 
              <a href="#">@v</a>
             }
         }</div>
    <div class="search-list-con">
        <dl class="search-list">
            @if (Model != null&& Model.Count > 0)
            {
                foreach (var viewModel in Model)
                {
                <dt><a href="@viewModel.Url" target="_blank">@MvcHtmlString.Create(viewModel.Title)</a>@viewModel.CreateTime</dt>
                <dd>@MvcHtmlString.Create(viewModel.Msg)</dd>
                }
            } 
              @Html.Pager(Model, new PagerOptions
 {
     PageIndexParameterName = "id",
     ShowPageIndexBox = true,
     FirstPageText = "首页",
     PrevPageText = "上一页",
     NextPageText = "下一页",
     LastPageText = "末页",
     PageIndexBoxType = PageIndexBoxType.TextBox,
     PageIndexBoxWrapperFormatString = "请输入页数{0}",
     GoButtonText = "转到"
 })
     <br />
     >>分页 共有 @(Model==null? 0: Model.TotalItemCount) 篇文章 @(Model==null?0:Model.CurrentPageIndex)/@(Model==null?0:Model.TotalPageCount)
        </dl>
    </div>
    <div>@ViewData["ShowInfo"]</div>
}
<script type="text/javascript">
    $(function () {
        $("#divKeyWords a").click(function () {
            $("#txtSearch").val($(this).html());
            $("#btnSearch").click();
        });
        getKeyWordsList("txtSearch");
        $("#isOr").click(function () {
            if ($(this).attr("checked") == "checked") {
                $("#hidfIsOr").val(true);
            }
            else {
                $("#hidfIsOr").val(false);
            }
        });
        if ($("#hidfIsOr").val() == "true") {
            $("input[type='checkbox']").prop("checked", true);
        }
    });
    //自动加载搜索列表
    function getKeyWordsList(txt) {
        if (txt == undefined || txt == "")
            return;
        $("#" + txt).autocomplete({
            source: "/LastSearch/GetKeyWordsList",
            minLength: 1
        });
    }
</script>
<script src="~/lib/Autocomplete/js/jquery-ui-1.8.17.custom.min.js"></script>

View Code

然后,各位看官请再看LastSearch调控器中的方法:

图片 8图片 9

 public class LastSearchController : Controller
    {
        //
        // GET: /LastSearch/

        string indexPath = System.Configuration.ConfigurationManager.AppSettings["lucenedir"];
        private SearchDemoContext db = new SearchDemoContext();

              public ActionResult Index(string txtSearch, bool? hidfIsOr, int id = 1)
        {
            PagedList<SearchResult> list = null;
            if (!string.IsNullOrEmpty(txtSearch))//如果点击的是查询按钮
            {
                //list = Search(txtSearch);
                list = (hidfIsOr == null || hidfIsOr.Value == false) ? OrSearch(txtSearch, id) : AndSearch(txtSearch, id);
            }
            var keyWords = db.SearchTotal.OrderByDescending(a => a.SearchCounts).Select(x => x.KeyWords).Skip(0).Take(6).ToList();
            ViewBag.KeyWords = keyWords;
            return View(list);
        }
        //与查询
        PagedList<SearchResult> AndSearch(String kw, int pageNo, int pageLen = 4)
        {
            FSDirectory directory = FSDirectory.Open(new DirectoryInfo(indexPath), new NoLockFactory());
            IndexReader reader = IndexReader.Open(directory, true);
            IndexSearcher searcher = new IndexSearcher(reader);
            PhraseQuery query = new PhraseQuery();//查询条件
            PhraseQuery titleQuery = new PhraseQuery();//标题查询条件
            List<string> lstkw = LuceneHelper.PanGuSplitWord(kw);//对用户输入的搜索条件进行拆分。

            foreach (string word in lstkw)
            {
                query.Add(new Term("Content", word));//contains("Content",word)
                titleQuery.Add(new Term("Title", word));
            }
            query.SetSlop(100);//两个词的距离大于100(经验值)就不放入搜索结果,因为距离太远相关度就不高了

            BooleanQuery bq = new BooleanQuery();
            //Occur.Should 表示 Or , Must 表示 and 运算
            bq.Add(query, BooleanClause.Occur.SHOULD);
            bq.Add(titleQuery, BooleanClause.Occur.SHOULD);

            TopScoreDocCollector collector = TopScoreDocCollector.create(1000, true);//盛放查询结果的容器
            searcher.Search(bq, null, collector);//使用query这个查询条件进行搜索,搜索结果放入collector

            int recCount=collector.GetTotalHits();//总的结果条数
            ScoreDoc[] docs = collector.TopDocs((pageNo - 1) * pageLen, pageNo*pageLen).scoreDocs;//从查询结果中取出第m条到第n条的数据

            List<SearchResult> list = new List<SearchResult>();
            string msg = string.Empty;
            string title = string.Empty;

            for (int i = 0; i < docs.Length; i++)//遍历查询结果
            {
                int docId = docs[i].doc;//拿到文档的id,因为Document可能非常占内存(思考DataSet和DataReader的区别)
                //所以查询结果中只有id,具体内容需要二次查询
                Document doc = searcher.Doc(docId);//根据id查询内容。放进去的是Document,查出来的还是Document
                SearchResult result = new SearchResult();
                result.Id = Convert.ToInt32(doc.Get("Id"));
                msg = doc.Get("Content");//只有 Field.Store.YES的字段才能用Get查出来
                result.Msg = LuceneHelper.CreateHightLight(kw, msg);//将搜索的关键字高亮显示。
                title = doc.Get("Title");
                foreach (string word in lstkw)
                {
                    title=title.Replace(word,""+word+"");
                }
                //result.Title=LuceneHelper.CreateHightLight(kw, title);
                result.Title = title;
                result.CreateTime = Convert.ToDateTime(doc.Get("CreateTime"));
                result.Url = "/Article/Details?Id=" + result.Id + "&kw=" + kw;
                list.Add(result);
            }
            //先将搜索的词插入到明细表。
            SearchDetail _SearchDetail = new SearchDetail { Id = Guid.NewGuid(), KeyWords = kw, SearchDateTime = DateTime.Now };
            db.SearchDetail.Add(_SearchDetail);
            int r = db.SaveChanges();

            PagedList<SearchResult> lst = new PagedList<SearchResult>(list, pageNo, pageLen, recCount);
            lst.TotalItemCount = recCount;
            lst.CurrentPageIndex = pageNo;

            return lst;
        }
        //或查询
        PagedList<SearchResult> OrSearch(String kw, int pageNo, int pageLen = 4)
        {
            FSDirectory directory = FSDirectory.Open(new DirectoryInfo(indexPath), new NoLockFactory());
            IndexReader reader = IndexReader.Open(directory, true);
            IndexSearcher searcher = new IndexSearcher(reader);
            List<PhraseQuery> lstQuery = new List<PhraseQuery>();
            List<string> lstkw = LuceneHelper.PanGuSplitWord(kw);//对用户输入的搜索条件进行拆分。

            foreach (string word in lstkw)
            {
                PhraseQuery query = new PhraseQuery();//查询条件
                query.SetSlop(100);//两个词的距离大于100(经验值)就不放入搜索结果,因为距离太远相关度就不高了
                query.Add(new Term("Content", word));//contains("Content",word)

                PhraseQuery titleQuery = new PhraseQuery();//查询条件
                titleQuery.Add(new Term("Title", word));

                lstQuery.Add(query);
                lstQuery.Add(titleQuery);
            }

            BooleanQuery bq = new BooleanQuery();
            foreach (var v in lstQuery)
            {
                //Occur.Should 表示 Or , Must 表示 and 运算
                bq.Add(v, BooleanClause.Occur.SHOULD);
            }
            TopScoreDocCollector collector = TopScoreDocCollector.create(1000, true);//盛放查询结果的容器
            searcher.Search(bq, null, collector);//使用query这个查询条件进行搜索,搜索结果放入collector

            int recCount = collector.GetTotalHits();//总的结果条数
            ScoreDoc[] docs = collector.TopDocs((pageNo - 1) * pageLen, pageNo * pageLen).scoreDocs;//从查询结果中取出第m条到第n条的数据

            List<SearchResult> list = new List<SearchResult>();
            string msg = string.Empty;
            string title = string.Empty;

            for (int i = 0; i < docs.Length; i++)//遍历查询结果
            {
                int docId = docs[i].doc;//拿到文档的id,因为Document可能非常占内存(思考DataSet和DataReader的区别)
                //所以查询结果中只有id,具体内容需要二次查询
                Document doc = searcher.Doc(docId);//根据id查询内容。放进去的是Document,查出来的还是Document
                SearchResult result = new SearchResult();
                result.Id = Convert.ToInt32(doc.Get("Id"));
                msg = doc.Get("Content");//只有 Field.Store.YES的字段才能用Get查出来
                result.Msg = LuceneHelper.CreateHightLight(kw, msg);//将搜索的关键字高亮显示。
                title = doc.Get("Title");
                foreach (string word in lstkw)
                {
                    title = title.Replace(word, "" + word + "");
                }
                //result.Title=LuceneHelper.CreateHightLight(kw, title);
                result.Title = title;
                result.CreateTime = Convert.ToDateTime(doc.Get("CreateTime"));
                result.Url = "/Article/Details?Id=" + result.Id + "&kw=" + kw;
                list.Add(result);
            }
            //先将搜索的词插入到明细表。
            SearchDetail _SearchDetail = new SearchDetail { Id = Guid.NewGuid(), KeyWords = kw, SearchDateTime = DateTime.Now };
            db.SearchDetail.Add(_SearchDetail);
            int r = db.SaveChanges();

            PagedList<SearchResult> lst = new PagedList<SearchResult>(list, pageNo, pageLen, recCount);
            lst.TotalItemCount = recCount;
            lst.CurrentPageIndex = pageNo;

            return lst;
        }

        /// <summary>
        /// 获取客户列表 模糊查询
        /// </summary>
        /// <param name="term"></param>
        /// <returns></returns>
        public string GetKeyWordsList(string term)
        {
            if (string.IsNullOrWhiteSpace(term))
                return null;

            var list = new KeyWordsTotalService().GetSearchMsg(term);
            //序列化对象
            //尽量不要用JavaScriptSerializer,为什么?性能差,完全可用Newtonsoft.Json来代替
            //System.Web.Script.Serialization.JavaScriptSerializer js = new System.Web.Script.Serialization.JavaScriptSerializer();
            //return js.Serialize(list.ToArray());
            return JsonConvert.SerializeObject(list.ToArray());
        }

View Code

现今,站内找寻的功底用均已做到。

相关文章