在工作中,我们平常会遭遇过对UCD团队内部成员开展KPI绩效考核,咋样考评设计师能力,并且付诸合理的KPI考核标准呐?

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

像www.verycd.com、新浪、Tmall、京东都有落实站内搜索效能,站内搜索无论在性质和用户体验上都丰富不易,本节,通过接纳Lucene.Net来实现站内搜索。

示范效果预览如下图10-22~10-24所示。

统计 1

图10-22

 统计 2

图10-23

 统计 3

图10-24

在10.4节,已经完结了搜寻的第一个版本,不过还有很多地方需要优化。比如说,我要总计紧要词搜索的频率高的词,也即热词,以及像百度查寻这样,在输入关键字后,会自行把搜索相关的热词自动以下拉列表的格局带出去。还有诸如搜索结果分页,查看著作明细等。

对此专业性的集团来说,首先要制订出合理的符合本机关的KPI考核目标,即设计师需要所有的力量模型。以此视作在店堂侧面考核计划上下的一个正经。

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)在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)修改SearchDemoContext类,新增了性能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
运行。操作完成将来,会在数据库SearchDemo中多新建两张表SearchTotals、SearchDetails,而原先的Articles表保持不变。如图10-20所示。

 统计 4

图10-20

(4)保存搜索记录

用户在历次搜寻的时候,要把搜索记录存入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)定时更新SearchTotals表记录

见状这种定时任务操作,这里可以行使Quartz.Net框架,为了便于,我把Quartz.Net的Job寄宿在控制台程序中,而实在工作中,我则更倾向于将其寄宿在Windows服务中。假诺有必要,可以把这多少个定时更新SearchTotals表记录的顺序部署到独门的服务器,那样可以减轻Web服务器的压力。

  1. 新建控制台程序QuartzNet,添加Quartz.dll和Common.Logging.dll的程序集引用,这里运用Database
    First的主意,添加ADO.NET实体数据模型,把表SearchTotals、SearchDetails添加进来。

2.添加KeyWordsTotal瑟维斯(Service).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,然后我们去开展检索操作,这样后台就定期的变型了搜寻总计记录。

标准设计师能力模型的优势

1.
对此设计师来说,从初级、中级、高级、专家级等,有一个较强烈的职业规划。同时依据实际的力量模型(设计矩阵),来寻找自己欠缺的能力来完善和弥补。对增长自我力量的目的性会更强。

2.
对于公司KPI考核来说,制定了规范的考核标准,避免了人为因素造成的考核失真,不公平的现象会尽量避免。就好比拿一把尺子去权衡总比凭感觉去测量更加精准的道理一样。

上面推荐Medium上的一篇咋样裁判产品设计师的好坏文中中的部分内容,来详细的为我们介绍设计师能力模型搭建。

急需提出的是,不同公司UCD队搭建不同,设计师定位及任务分开也都会有出入,所以列表中的能力模型只是全体的通用模型,各自团队需要在此基础上进展连续调整。

10.5.2 热门搜索

本着视觉UI设计师能力模型(每个能力模型从Level 0-3共4个阶段)

  1. UI用户界面视觉设计

  2. 互相原型

  3. 品牌设计

  4. 插画手绘能力(视觉能力)

  5. 互动动功用力(MG动画)

  6. 民用博客、项目经验总括(理论实践与方法论结合)

视觉设计师能力模型

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>

UX用户体验设计师能力模型

  1. 用户调研与分析(用户画像、用户体验地图等)

2.用户数量解析 (能应用相关数据总结工具,举行数量解析和整理)

3. 可用性测试与分析(遵照评估和概括调查结果,将迎刃而解方案提供给品种有关人士及利益相关者)

UX用户体验设计师能力模型

10.5.2.2 搜索下拉框

此处我引入一个第三方js框架Autocomplete,它能在文本框中输入文字的时候,自动从后台抓去数据下拉列表。

云盘中本人提供了Autocomplete.rar,将其解压,然后拷贝到SearchDemo项目中的lib目录下。

在SearchDemo项目中的KeyWordsTotal瑟维斯(Service).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>

前端工程师能力模型

在国内UCD团队中,大部分业已不安装前端工程师岗位,而细分到FE开发协会中。

前者工程师能力模型

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

通用工作力量考核指数

  1. 版本控制

  2. 自动化构建

  3. 代码可读性

  4. 代码协会架构

  5. HTML&CSS

  6. 内容管理系列

  7. JavaScript语言

  8. HTML/CSS

9.  JavaScript、Framework、Knowledge语言(依照公司确定)

10.支出测试

通用工作力量考核

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

由来,站内搜索的基本效能均已形成。

其实测评案例

  1. 关系能力

  2. 决策力

3. 开发语言熟悉度(根据不同公司情况)

  1. 品种经验/项目输出

  2. 计划社交、社群(是否经常浏览有关专业设计平台)

  3. 个人简历随笔集

私家力量模型评分_案例彰显

以上设计师能力模型搭建援引自medium著作,其实针对国内UCD团队举办KPI考核过程中,很多东西都是亟需转变。按照集团行业、团队架构、设计师职责分开等居多要素开展动态调整。通过本译文,只是想传达给大家在设计师能力模型搭建过程中,需要考虑的部分关键点。

延伸阅读:

What is a good product designer?

https://blog.usejournal.com/what-is-a-good-product-designer-76a5235b6dbf

Product Designer Competency
Matrix

https://docs.google.com/spreadsheets/d/1pm5NLpKmiSmYI3G65GJ8QwO6PgShHyrfWvUbJT5Dmlg/edit\#gid=0