csdn上有很多自己的博客,不过现在用户体验太差,准备迁移到hexo上,本工具一键迁移项目。
- 生成jar包
mvn clean package
- 生成json文件
java -jar csdn-blog-export-0.0.1-SNAPSHOT.jar d --cookie=你的cookie --filePath=存储文件路径
抓包完毕会生成一个你定义的文件,文件包含了所有的文章信息。格式例如为:
[
{
"data":{
"date":"2018年11月20日 21:50:04",
"articleedittype":1,
"private":0,
"channel":"2",
"description":"深入推荐引擎相关算法 - 协同过滤\n集体智慧和协同过滤\n什么是集体智慧\n集体智慧 (Collective Intelligence) 并不是 Web2.0 时代特有的,只是在 Web2.0 时代,大家在 Web 应用中利用集体智慧构建更加有趣的应用或者得到更好的用户体验。集体智慧是指在大量的人群的行为和数据中收集答案,帮助你对整个人群得到统计意义上的结论,这些结论是我们在单个个体上无法得到的,它往...",
"title":"深入推荐引擎相关算法 - 协同过滤",
"type":"repost",
"content":"<h1><a id=\"___0\"></a>深入推荐引擎相关算法 - 协同过滤</h1>\n<h2><a id=\"_2\"></a>集体智慧和协同过滤</h2>\n<h3><a id=\"_4\"></a>什么是集体智慧</h3>\n<p>集体智慧 (Collective Intelligence) 并不是 Web2.0 时代特有的,只是在 Web2.0 时代,大家在 Web 应用中利用集体智慧构建更加有趣的应用或者得到更好的用户体验。集体智慧是指在大量的人群的行为和数据中收集答案,帮助你对整个人群得到统计意义上的结论,这些结论是我们在单个个体上无法得到的,它往往是某种趋势或者人群中共性的部分。</p>\n<p>Wikipedia 和 Google 是两个典型的利用集体智慧的 Web 2.0 应用:</p>\n<ul>\n<li>Wikipedia 是一个知识管理的百科全书,相对于传统的由领域专家编辑的百科全书,Wikipedia 允许最终用户贡献知识,随着参与人数的增多,Wikipedia 变成了涵盖各个领域的一本无比全面的知识库。也许有人会质疑它的权威性,但如果你从另一个侧面想这个问题,也许就可以迎刃而解。在发行一本书时,作者虽然是权威,但难免还有一些错误,然后通过一版一版的改版,书的内容越来越完善。而在 Wikipedia 上,这种改版和修正被变为每个人都可以做的事情,任何人发现错误或者不完善都可以贡献他们的想法,即便某些信息是错误的,但它一定也会尽快的被其他人纠正过来。从一个宏观的角度看,整个系统在按照一个良性循环的轨迹不断完善,这也正是集体智慧的魅力。</li>\n<li>Google:目前最流行的搜索引擎,与 Wikipedia 不同,它没有要求用户显式的贡献,但仔细想想 Google 最核心的 PageRank 的思想,它利用了 Web 页面之间的关系,将多少其他页面链接到当前页面的数目作为衡量当前页面重要与否的标准;如果这不好理解,那么你可以把它想象成一个选举的过程,每个 Web 页面都是一个投票者同时也是一个被投票者,PageRank 通过一定数目的迭代得到一个相对稳定的评分。Google 其实利用了现在 Internet 上所有 Web 页面上链接的集体智慧,找到哪些页面是重要的。</li>\n</ul>\n<h3><a id=\"_13\"></a>什么是协同过滤</h3>\n<p>协同过滤是利用集体智慧的一个典型方法。要理解什么是协同过滤 (Collaborative Filtering, 简称 CF),首先想一个简单的问题,如果你现在想看个电影,但你不知道具体看哪部,你会怎么做?大部分的人会问问周围的朋友,看看最近有什么好看的电影推荐,而我们一般更倾向于从口味比较类似的朋友那里得到推荐。这就是协同过滤的核心思想。</p>\n<p>协同过滤一般是在海量的用户中发掘出一小部分和你品位比较类似的,在协同过滤中,这些用户成为邻居,然后根据他们喜欢的其他东西组织成一个排序的目录作为推荐给你。当然其中有一个核心的问题:</p>\n<ul>\n<li>如何确定一个用户是不是和你有相似的品位?</li>\n<li>如何将邻居们的喜好组织成一个排序的目录?</li>\n</ul>\n<p>协同过滤相对于集体智慧而言,它从一定程度上保留了个体的特征,就是你的品位偏好,所以它更多可以作为个性化推荐的算法思想。可以想象,这种推荐策略在 Web 2.0 的长尾中是很重要的,将大众流行的东西推荐给长尾中的人怎么可能得到好的效果,这也回到推荐系统的一个核心问题:了解你的用户,然后才能给出更好的推荐。</p>\n<h3><a id=\"_24\"></a>深入协同过滤的核心</h3>\n<p>前面作为背景知识,介绍了集体智慧和协同过滤的基本思想,这一节我们将深入分析协同过滤的原理,介绍基于协同过滤思想的多种推荐机制,优缺点和实用场景。</p>\n<p>首先,要实现协同过滤,需要一下几个步骤</p>\n<ul>\n<li>收集用户偏好</li>\n<li>找到相似的用户或物品</li>\n<li>计算推荐</li>\n</ul>\n<h3><a id=\"_34\"></a>收集用户偏好</h3>\n<p>要从用户的行为和偏好中发现规律,并基于此给予推荐,如何收集用户的偏好信息成为系统推荐效果最基础的决定因素。用户有很多方式向系统提供自己的偏好信息,而且不同的应用也可能大不相同,下面举例进行介绍:</p>\n<h5><a id=\"_1__38\"></a>表 1 用户行为和用户偏好</h5>\n\n<table>\n<thead>\n<tr>\n<th><strong>用户行为</strong></th>\n<th><strong>类型</strong></th>\n<th><strong>特征</strong></th>\n<th><strong>作用</strong></th>\n</tr>\n</thead>\n<tbody>\n<tr>\n<td>评分</td>\n<td>显式</td>\n<td>整数量化的偏好,可能的取值是 [0, n];n 一般取值为 5 或者是 10</td>\n<td>通过用户对物品的评分,可以精确的得到用户的偏好</td>\n</tr>\n<tr>\n<td>投票</td>\n<td>显式</td>\n<td>布尔量化的偏好,取值是 0 或 1</td>\n<td>通过用户对物品的投票,可以较精确的得到用户的偏好</td>\n</tr>\n<tr>\n<td>转发</td>\n<td>显式</td>\n<td>布尔量化的偏好,取值是 0 或 1</td>\n<td>通过用户对物品的投票,可以精确的得到用户的偏好。 如果是站内,同时可以推理得到被转发人的偏好(不精确)</td>\n</tr>\n<tr>\n<td>保存书签</td>\n<td>显示</td>\n<td>布尔量化的偏好,取值是 0 或 1</td>\n<td>通过用户对物品的投票,可以精确的得到用户的偏好。</td>\n</tr>\n<tr>\n<td>标记标签 (Tag)</td>\n<td>显示</td>\n<td>一些单词,需要对单词进行分析,得到偏好</td>\n<td>通过分析用户的标签,可以得到用户对项目的理解,同时可以分析出用户的情感:喜欢还是讨厌</td>\n</tr>\n<tr>\n<td>评论</td>\n<td>显示</td>\n<td>一段文字,需要进行文本分析,得到偏好</td>\n<td>通过分析用户的评论,可以得到用户的情感:喜欢还是讨厌</td>\n</tr>\n<tr>\n<td>点击流 ( 查看 )</td>\n<td>隐式</td>\n<td>一组用户的点击,用户对物品感兴趣,需要进行分析,得到偏好</td>\n<td>用户的点击一定程度上反映了用户的注意力,所以它也可以从一定程度上反映用户的喜好。</td>\n</tr>\n<tr>\n<td>页面停留时间</td>\n<td>隐式</td>\n<td>一组时间信息,噪音大,需要进行去噪,分析,得到偏好</td>\n<td>用户的页面停留时间一定程度上反映了用户的注意力和喜好,但噪音偏大,不好利用。</td>\n</tr>\n<tr>\n<td>购买</td>\n<td>隐式</td>\n<td>布尔量化的偏好,取值是 0 或 1</td>\n<td>用户的购买是很明确的说明这个项目它感兴趣。</td>\n</tr>\n</tbody>\n</table><p>以上列举的用户行为都是比较通用的,推荐引擎设计人员可以根据自己应用的特点添加特殊的用户行为,并用他们表示用户对物品的喜好。</p>\n<p>在一般应用中,我们提取的用户行为一般都多于一种,关于如何组合这些不同的用户行为,基本上有以下两种方式:</p>\n<ul>\n<li>将不同的行为分组:一般可以分为“查看”和“购买”等等,然后基于不同的行为,计算不同的用户 / 物品相似度。类似于当当网或者 Amazon 给出的“购买了该图书的人还购买了 …”,“查看了图书的人还查看了 …”</li>\n<li>根据不同行为反映用户喜好的程度将它们进行加权,得到用户对于物品的总体喜好。一般来说,显式的用户反馈比隐式的权值大,但比较稀疏,毕竟进行显示反馈的用户是少数;同时相对于“查看”,“购买”行为反映用户喜好的程度更大,但这也因应用而异。</li>\n</ul>\n<p>收集了用户行为数据,我们还需要对数据进行一定的预处理,其中最核心的工作就是:减噪和归一化。</p>\n<ul>\n<li>减噪:用户行为数据是用户在使用应用过程中产生的,它可能存在大量的噪音和用户的误操作,我们可以通过经典的数据挖掘算法过滤掉行为数据中的噪音,这样可以是我们的分析更加精确。</li>\n<li>归一化:如前面讲到的,在计算用户对物品的喜好程度时,可能需要对不同的行为数据进行加权。但可以想象,不同行为的数据取值可能相差很大,比如,用户的查看数据必然比购买数据大的多,如何将各个行为的数据统一在一个相同的取值范围中,从而使得加权求和得到的总体喜好更加精确,就需要我们进行归一化处理。最简单的归一化处理,就是将各类数据除以此类中的最大值,以保证归一化后的数据取值在 [0,1] 范围中。</li>\n</ul>\n<p>进行的预处理后,根据不同应用的行为分析方法,可以选择分组或者加权处理,之后我们可以得到一个用户偏好的二维矩阵,一维是用户列表,另一维是物品列表,值是用户对物品的偏好,一般是 [0,1] 或者 [-1, 1] 的浮点数值。</p>\n<h3><a id=\"_66\"></a>找到相似的用户或物品</h3>\n<p>当已经对用户行为进行分析得到用户喜好后,我们可以根据用户喜好计算相似用户和物品,然后基于相似用户或者物品进行推荐,这就是最典型的 CF 的两个分支:基于用户的 CF 和基于物品的 CF。这两种方法都需要计算相似度,下面我们先看看最基本的几种计算相似度的方法。</p>\n<p><strong>相似度的计算</strong></p>\n<p>关于相似度的计算,现有的几种基本方法都是基于向量(Vector)的,其实也就是计算两个向量的距离,距离越近相似度越大。在推荐的场景中,在用户 - 物品偏好的二维矩阵中,我们可以将一个用户对所有物品的偏好作为一个向量来计算用户之间的相似度,或者将所有用户对某个物品的偏好作为一个向量来计算物品之间的相似度。下面我们详细介绍几种常用的相似度计算方法:</p>\n<ul>\n<li>欧几里德距离(Euclidean Distance)</li>\n</ul>\n<p>最初用于计算欧几里德空间中两个点的距离,假设 x,y 是 n 维空间的两个点,它们之间的欧几里德距离是:</p>\n<p><img src=\"https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image003.gif\" alt=\"Figure xxx. Requires a heading\"></p>\n<p>可以看出,当 n=2 时,欧几里德距离就是平面上两个点的距离。欧几里德学习视频: <a href=\"https://www.yxgapp.com/video/c8426884-2b56-494f-a274-0aa3105503f1.html\">《机器学习Python实践》#15 欧几里得距离</a></p>\n<p>当用欧几里德距离表示相似度,一般采用以下公式进行转换:距离越小,相似度越大</p>\n<p><img src=\"https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image005.gif\" alt=\"Figure xxx. Requires a heading\"></p>\n<ul>\n<li>皮尔逊相关系数(Pearson Correlation Coefficient)</li>\n</ul>\n<p>皮尔逊相关系数一般用于计算两个定距变量间联系的紧密程度,它的取值在 [-1,+1] 之间。</p>\n<p><img src=\"https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image007.gif\" alt=\"Figure xxx. Requires a heading\"></p>\n<p>sx, sy是 x 和 y 的样品标准偏差。</p>\n<ul>\n<li>Cosine 相似度(Cosine Similarity)</li>\n</ul>\n<p>Cosine 相似度被广泛应用于计算文档数据的相似度:</p>\n<p><img src=\"https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image009.gif\" alt=\"Figure xxx. Requires a heading\"></p>\n<ul>\n<li>Tanimoto 系数(Tanimoto Coefficient)</li>\n</ul>\n<p>Tanimoto 系数也称为 Jaccard 系数,是 Cosine 相似度的扩展,也多用于计算文档数据的相似度:</p>\n<p><img src=\"https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image011.gif\" alt=\"Figure xxx. Requires a heading\"></p>\n<p><strong>相似邻居的计算</strong></p>\n<p>介绍完相似度的计算方法,下面我们看看如何根据相似度找到用户 - 物品的邻居,常用的挑选邻居的原则可以分为两类:图 1 给出了二维平面空间上点集的示意图。</p>\n<ul>\n<li>固定数量的邻居:K-neighborhoods 或者 Fix-size neighborhoods</li>\n</ul>\n<p>不论邻居的“远近”,只取最近的 K 个,作为其邻居。如图 1 中的 A,假设要计算点 1 的 5- 邻居,那么根据点之间的距离,我们取最近的 5 个点,分别是点 2,点 3,点 4,点 7 和点 5。但很明显我们可以看出,这种方法对于孤立点的计算效果不好,因为要取固定个数的邻居,当它附近没有足够多比较相似的点,就被迫取一些不太相似的点作为邻居,这样就影响了邻居相似的程度,比如图 1 中,点 1 和点 5 其实并不是很相似。</p>\n<ul>\n<li>基于相似度门槛的邻居:Threshold-based neighborhoods</li>\n</ul>\n<p>与计算固定数量的邻居的原则不同,基于相似度门槛的邻居计算是对邻居的远近进行最大值的限制,落在以当前点为中心,距离为 K 的区域中的所有点都作为当前点的邻居,这种方法计算得到的邻居个数不确定,但相似度不会出现较大的误差。如图 1 中的 B,从点 1 出发,计算相似度在 K 内的邻居,得到点 2,点 3,点 4 和点 7,这种方法计算出的邻居的相似度程度比前一种优,尤其是对孤立点的处理。</p>\n<h5><a id=\"_1_124\"></a>图 1.相似邻居计算示意图</h5>\n<p><img src=\"https://ws4.sinaimg.cn/large/006tNbRwly1fxev00vhkxg30ej08iwef.gif\" alt=\"\"></p>\n<h3><a id=\"_128\"></a>计算推荐</h3>\n<p>经过前期的计算已经得到了相邻用户和相邻物品,下面介绍如何基于这些信息为用户进行推荐。本系列的上一篇综述文章已经简要介绍过基于协同过滤的推荐算法可以分为基于用户的 CF 和基于物品的 CF,下面我们深入这两种方法的计算方法,使用场景和优缺点。</p>\n<p><strong>基于用户的 CF(User CF)</strong></p>\n<p>基于用户的 CF 的基本思想相当简单,基于用户对物品的偏好找到相邻邻居用户,然后将邻居用户喜欢的推荐给当前用户。计算上,就是将一个用户对所有物品的偏好作为一个向量来计算用户之间的相似度,找到 K 邻居后,根据邻居的相似度权重以及他们对物品的偏好,预测当前用户没有偏好的未涉及物品,计算得到一个排序的物品列表作为推荐。图 2 给出了一个例子,对于用户 A,根据用户的历史偏好,这里只计算得到一个邻居 - 用户 C,然后将用户 C 喜欢的物品 D 推荐给用户 A。</p>\n<h5><a id=\"_2_CF__136\"></a>图 2.基于用户的 CF 的基本原理</h5>\n<p><img src=\"https://ws3.sinaimg.cn/large/006tNbRwly1fxev0k5q29g30aw08wglm.gif\" alt=\"\"></p>\n<p><strong>基于物品的 CF(Item CF)</strong></p>\n<p>基于物品的 CF 的原理和基于用户的 CF 类似,只是在计算邻居时采用物品本身,而不是从用户的角度,即基于用户对物品的偏好找到相似的物品,然后根据用户的历史偏好,推荐相似的物品给他。从计算的角度看,就是将所有用户对某个物品的偏好作为一个向量来计算物品之间的相似度,得到物品的相似物品后,根据用户历史的偏好预测当前用户还没有表示偏好的物品,计算得到一个排序的物品列表作为推荐。图 3 给出了一个例子,对于物品 A,根据所有用户的历史偏好,喜欢物品 A 的用户都喜欢物品 C,得出物品 A 和物品 C 比较相似,而用户 C 喜欢物品 A,那么可以推断出用户 C 可能也喜欢物品 C。</p>\n<h5><a id=\"_3_CF__144\"></a>图 3.基于物品的 CF 的基本原理</h5>\n<p><img src=\"https://ws2.sinaimg.cn/large/006tNbRwly1fxev0tb00rg30aa099glm.gif\" alt=\"†\"></p>\n<p><strong>User CF vs. Item CF</strong></p>\n<p>前面介绍了 User CF 和 Item CF 的基本原理,下面我们分几个不同的角度深入看看它们各自的优缺点和适用场景:</p>\n<ul>\n<li>计算复杂度</li>\n</ul>\n<p>Item CF 和 User CF 是基于协同过滤推荐的两个最基本的算法,User CF 是很早以前就提出来了,Item CF 是从 Amazon 的论文和专利发表之后(2001 年左右)开始流行,大家都觉得 Item CF 从性能和复杂度上比 User CF 更优,其中的一个主要原因就是对于一个在线网站,用户的数量往往大大超过物品的数量,同时物品的数据相对稳定,因此计算物品的相似度不但计算量较小,同时也不必频繁更新。但我们往往忽略了这种情况只适应于提供商品的电子商务网站,对于新闻,博客或者微内容的推荐系统,情况往往是相反的,物品的数量是海量的,同时也是更新频繁的,所以单从复杂度的角度,这两个算法在不同的系统中各有优势,推荐引擎的设计者需要根据自己应用的特点选择更加合适的算法。</p>\n<ul>\n<li>适用场景</li>\n</ul>\n<p>在非社交网络的网站中,内容内在的联系是很重要的推荐原则,它比基于相似用户的推荐原则更加有效。比如在购书网站上,当你看一本书的时候,推荐引擎会给你推荐相关的书籍,这个推荐的重要性远远超过了网站首页对该用户的综合推荐。可以看到,在这种情况下,Item CF 的推荐成为了引导用户浏览的重要手段。同时 Item CF 便于为推荐做出解释,在一个非社交网络的网站中,给某个用户推荐一本书,同时给出的解释是某某和你有相似兴趣的人也看了这本书,这很难让用户信服,因为用户可能根本不认识那个人;但如果解释说是因为这本书和你以前看的某本书相似,用户可能就觉得合理而采纳了此推荐。</p>\n<p>相反的,在现今很流行的社交网络站点中,User CF 是一个更不错的选择,User CF 加上社会网络信息,可以增加用户对推荐解释的信服程度。</p>\n<ul>\n<li>推荐多样性和精度</li>\n</ul>\n<p>研究推荐引擎的学者们在相同的数据集合上分别用 User CF 和 Item CF 计算推荐结果,发现推荐列表中,只有 50% 是一样的,还有 50% 完全不同。但是这两个算法确有相似的精度,所以可以说,这两个算法是很互补的。</p>\n<p>关于推荐的多样性,有两种度量方法:</p>\n<p>第一种度量方法是从单个用户的角度度量,就是说给定一个用户,查看系统给出的推荐列表是否多样,也就是要比较推荐列表中的物品之间两两的相似度,不难想到,对这种度量方法,Item CF 的多样性显然不如 User CF 的好,因为 Item CF 的推荐就是和以前看的东西最相似的。</p>\n<p>第二种度量方法是考虑系统的多样性,也被称为覆盖率 (Coverage),它是指一个推荐系统是否能够提供给所有用户丰富的选择。在这种指标下,Item CF 的多样性要远远好于 User CF, 因为 User CF 总是倾向于推荐热门的,从另一个侧面看,也就是说,Item CF 的推荐有很好的新颖性,很擅长推荐长尾里的物品。所以,尽管大多数情况,Item CF 的精度略小于 User CF, 但如果考虑多样性,Item CF 却比 User CF 好很多。</p>\n<p>如果你对推荐的多样性还心存疑惑,那么下面我们再举个实例看看 User CF 和 Item CF 的多样性到底有什么差别。首先,假设每个用户兴趣爱好都是广泛的,喜欢好几个领域的东西,不过每个用户肯定也有一个主要的领域,对这个领域会比其他领域更加关心。给定一个用户,假设他喜欢 3 个领域 A,B,C,A 是他喜欢的主要领域,这个时候我们来看 User CF 和 Item CF 倾向于做出什么推荐:如果用 User CF, 它会将 A,B,C 三个领域中比较热门的东西推荐给用户;而如果用 ItemCF,它会基本上只推荐 A 领域的东西给用户。所以我们看到因为 User CF 只推荐热门的,所以它在推荐长尾里项目方面的能力不足;而 Item CF 只推荐 A 领域给用户,这样他有限的推荐列表中就可能包含了一定数量的不热门的长尾物品,同时 Item CF 的推荐对这个用户而言,显然多样性不足。但是对整个系统而言,因为不同的用户的主要兴趣点不同,所以系统的覆盖率会比较好。</p>\n<p>从上面的分析,可以很清晰的看到,这两种推荐都有其合理性,但都不是最好的选择,因此他们的精度也会有损失。其实对这类系统的最好选择是,如果系统给这个用户推荐 30 个物品,既不是每个领域挑选 10 个最热门的给他,也不是推荐 30 个 A 领域的给他,而是比如推荐 15 个 A 领域的给他,剩下的 15 个从 B,C 中选择。所以结合 User CF 和 Item CF 是最优的选择,结合的基本原则就是当采用 Item CF 导致系统对个人推荐的多样性不足时,我们通过加入 User CF 增加个人推荐的多样性,从而提高精度,而当因为采用 User CF 而使系统的整体多样性不足时,我们可以通过加入 Item CF 增加整体的多样性,同样同样可以提高推荐的精度。</p>\n<ul>\n<li>用户对推荐算法的适应度</li>\n</ul>\n<p>前面我们大部分都是从推荐引擎的角度考虑哪个算法更优,但其实我们更多的应该考虑作为推荐引擎的最终使用者 – 应用用户对推荐算法的适应度。</p>\n<p>对于 User CF,推荐的原则是假设用户会喜欢那些和他有相同喜好的用户喜欢的东西,但如果一个用户没有相同喜好的朋友,那 User CF 的算法的效果就会很差,所以一个用户对的 CF 算法的适应度是和他有多少共同喜好用户成正比的。</p>\n<p>Item CF 算法也有一个基本假设,就是用户会喜欢和他以前喜欢的东西相似的东西,那么我们可以计算一个用户喜欢的物品的自相似度。一个用户喜欢物品的自相似度大,就说明他喜欢的东西都是比较相似的,也就是说他比较符合 Item CF 方法的基本假设,那么他对 Item CF 的适应度自然比较好;反之,如果自相似度小,就说明这个用户的喜好习惯并不满足 Item CF 方法的基本假设,那么对于这种用户,用 Item CF 方法做出好的推荐的可能性非常低。</p>\n<p>通过以上的介绍,相信大家已经对协同过滤推荐的各种方法,原则,特点和适用场景有深入的了解,下面我们就进入实战阶段,重点介绍如何基于 Apache Mahout 实现协同过滤推荐算法。</p>\n<h2><a id=\"_Apache_Mahout__186\"></a>基于 Apache Mahout 实现高效的协同过滤推荐</h2>\n<p>Apache Mahout 是 Apache Software Foundation (ASF) 旗下的一个开源项目,提供一些可扩展的机器学习领域经典算法的实现,旨在帮助开发人员更加方便快捷地创建智能应用程序,并且,在 Mahout 的最近版本中还加入了对 Apache Hadoop 的支持,使这些算法可以更高效的运行在云计算环境中。</p>\n<p>关于 Apache Mahout 的安装和配置请参考《基于 Apache Mahout 构建社会化推荐引擎》,它是笔者 09 年发表的一篇关于基于 Mahout 实现推荐引擎的 developerWorks 文章,其中详细介绍了 Mahout 的安装步骤,并给出一个简单的电影推荐引擎的例子。</p>\n<p>Apache Mahout 中提供的一个协同过滤算法的高效实现,它是一个基于 Java 实现的可扩展的,高效的推荐引擎。图 4 给出了 Apache Mahout 中协同过滤推荐实现的组件图,下面我们逐步深入介绍各个部分。</p>\n<h5><a id=\"_4_194\"></a>图 4.组件图</h5>\n<p><img src=\"https://ws3.sinaimg.cn/large/006tNbRwly1fxev2pr7jeg30cs09zglj.gif\" alt=\"\"></p>\n<h3><a id=\"Data_Model_198\"></a>数据表示:Data Model</h3>\n<p><strong>Preference</strong></p>\n<p>基于协同过滤的推荐引擎的输入是用户的历史偏好信息,在 Mahout 里它被建模为 Preference(接口),一个 Preference 就是一个简单的三元组 < 用户 ID, 物品 ID, 用户偏好 >,它的实现类是 GenericPreference,可以通过以下语句创建一个 GenericPreference。</p>\n<p>GenericPreference preference = new GenericPreference(123, 456, 3.0f);</p>\n<p>这其中, 123 是用户 ID,long 型;456 是物品 ID,long 型;3.0f 是用户偏好,float 型。从这个例子我们可以看出,单单一个 GenericPreference 的数据就占用 20 bytes,所以你会发现如果只简单实用数组 Array 加载用户偏好数据,必然占用大量的内存,Mahout 在这方面做了一些优化,它创建了 PreferenceArray(接口)保存一组用户偏好数据,为了优化性能,Mahout 给出了两个实现类,GenericUserPreferenceArray 和 GenericItemPreferenceArray,分别按照用户和物品本身对用户偏好进行组装,这样就可以压缩用户 ID 或者物品 ID 的空间。下面清单 1 的代码以 GenericUserPreferenceArray 为例,展示了如何创建和使用一个 PreferenceArray。</p>\n<h5><a id=\"_1__PreferenceArray_208\"></a>清单 1. 创建和使用 PreferenceArray</h5>\n<pre><code class=\"prism language-java\">PreferenceArray userPref <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">GenericUserPreferenceArray</span><span class=\"token punctuation\">(</span><span class=\"token number\">2</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> <span class=\"token comment\">//size = 2 </span>\n \nuserPref<span class=\"token punctuation\">.</span><span class=\"token function\">setUserID</span><span class=\"token punctuation\">(</span><span class=\"token number\">0</span><span class=\"token punctuation\">,</span> <span class=\"token number\">1</span>L<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n \nuserPref<span class=\"token punctuation\">.</span><span class=\"token function\">setItemID</span><span class=\"token punctuation\">(</span><span class=\"token number\">0</span><span class=\"token punctuation\">,</span> <span class=\"token number\">101</span>L<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> <span class=\"token comment\">//<1L, 101L, 2.0f> </span>\nuserPref<span class=\"token punctuation\">.</span><span class=\"token function\">setValue</span><span class=\"token punctuation\">(</span><span class=\"token number\">0</span><span class=\"token punctuation\">,</span> <span class=\"token number\">2.0f</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nuserPref<span class=\"token punctuation\">.</span><span class=\"token function\">setItemID</span><span class=\"token punctuation\">(</span><span class=\"token number\">1</span><span class=\"token punctuation\">,</span> <span class=\"token number\">102</span>L<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> <span class=\"token comment\">//<1L, 102L, 4.0f> </span>\nuserPref<span class=\"token punctuation\">.</span><span class=\"token function\">setValue</span><span class=\"token punctuation\">(</span><span class=\"token number\">1</span><span class=\"token punctuation\">,</span> <span class=\"token number\">4.0f</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n \nPreference pref <span class=\"token operator\">=</span> userPref<span class=\"token punctuation\">.</span><span class=\"token function\">get</span><span class=\"token punctuation\">(</span><span class=\"token number\">1</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> <span class=\"token comment\">//<1L, 102L, 4.0f></span>\n</code></pre>\n<p>为了提高性能 Mahout 还构建了自己的 HashMap 和 Set:FastByIDMap 和 FastIDSet,有兴趣的朋友可以参考 Mahout 官方说明。</p>\n<p><strong>DataModel</strong></p>\n<p>Mahout 的推荐引擎实际接受的输入是 DataModel,它是对用户偏好数据的压缩表示,通过创建内存版 DataModel 的语句我们可以看出:</p>\n<p>DataModel model = new GenericDataModel(FastByIDMap map);</p>\n<p>他保存在一个按照用户 ID 或者物品 ID 进行散列的 PreferenceArray,而 PreferenceArray 中对应保存着这个用户 ID 或者物品 ID 的所有用户偏好信息。</p>\n<p>DataModel 是用户喜好信息的抽象接口,它的具体实现支持从任意类型的数据源抽取用户喜好信息,具体实现包括内存版的 GenericDataModel,支持文件读取的 FileDataModel 和支持数据库读取的 JDBCDataModel,下面我们看看如何创建各种 DataModel。</p>\n<h5><a id=\"_2__DataModel_235\"></a>清单 2. 创建各种 DataModel</h5>\n<pre><code class=\"prism language-java\"><span class=\"token comment\">//In-memory DataModel - GenericDataModel </span>\n FastByIDMap<span class=\"token generics function\"><span class=\"token punctuation\"><</span>PreferenceArray<span class=\"token punctuation\">></span></span> preferences <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">FastByIDMap</span><span class=\"token generics function\"><span class=\"token punctuation\"><</span>PreferenceArray<span class=\"token punctuation\">></span></span><span class=\"token punctuation\">(</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n \n PreferenceArray prefsForUser1 <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">GenericUserPreferenceArray</span><span class=\"token punctuation\">(</span><span class=\"token number\">10</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n prefsForUser1<span class=\"token punctuation\">.</span><span class=\"token function\">setUserID</span><span class=\"token punctuation\">(</span><span class=\"token number\">0</span><span class=\"token punctuation\">,</span> <span class=\"token number\">1</span>L<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n prefsForUser1<span class=\"token punctuation\">.</span><span class=\"token function\">setItemID</span><span class=\"token punctuation\">(</span><span class=\"token number\">0</span><span class=\"token punctuation\">,</span> <span class=\"token number\">101</span>L<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n prefsForUser1<span class=\"token punctuation\">.</span><span class=\"token function\">setValue</span><span class=\"token punctuation\">(</span><span class=\"token number\">0</span><span class=\"token punctuation\">,</span> <span class=\"token number\">3.0f</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n prefsForUser1<span class=\"token punctuation\">.</span><span class=\"token function\">setItemID</span><span class=\"token punctuation\">(</span><span class=\"token number\">1</span><span class=\"token punctuation\">,</span> <span class=\"token number\">102</span>L<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n prefsForUser1<span class=\"token punctuation\">.</span><span class=\"token function\">setValue</span><span class=\"token punctuation\">(</span><span class=\"token number\">1</span><span class=\"token punctuation\">,</span> <span class=\"token number\">4.5f</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n… <span class=\"token punctuation\">(</span><span class=\"token number\">8</span> more<span class=\"token punctuation\">)</span> \n preferences<span class=\"token punctuation\">.</span><span class=\"token function\">put</span><span class=\"token punctuation\">(</span><span class=\"token number\">1</span>L<span class=\"token punctuation\">,</span> prefsForUser1<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> <span class=\"token comment\">//use userID as the key </span>\n… <span class=\"token punctuation\">(</span>more users<span class=\"token punctuation\">)</span> \n \n DataModel model <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">GenericDataModel</span><span class=\"token punctuation\">(</span>preferences<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n \n <span class=\"token comment\">//File-based DataModel - FileDataModel </span>\n DataModel dataModel <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">FileDataModel</span><span class=\"token punctuation\">(</span><span class=\"token keyword\">new</span> <span class=\"token class-name\">File</span><span class=\"token punctuation\">(</span><span class=\"token string\">\"preferences.csv\"</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n \n <span class=\"token comment\">//Database-based DataModel - MySQLJDBCDataModel </span>\n MysqlDataSource dataSource <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">MysqlDataSource</span><span class=\"token punctuation\">(</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n dataSource<span class=\"token punctuation\">.</span><span class=\"token function\">setServerName</span><span class=\"token punctuation\">(</span><span class=\"token string\">\"my_user\"</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n dataSource<span class=\"token punctuation\">.</span><span class=\"token function\">setUser</span><span class=\"token punctuation\">(</span><span class=\"token string\">\"my_password\"</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n dataSource<span class=\"token punctuation\">.</span><span class=\"token function\">setPassword</span><span class=\"token punctuation\">(</span><span class=\"token string\">\"my_database_host\"</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n JDBCDataModel dataModel <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">MySQLJDBCDataModel</span><span class=\"token punctuation\">(</span>dataSource<span class=\"token punctuation\">,</span> <span class=\"token string\">\"my_prefs_table\"</span><span class=\"token punctuation\">,</span> \n <span class=\"token string\">\"my_user_column\"</span><span class=\"token punctuation\">,</span> <span class=\"token string\">\"my_item_column\"</span><span class=\"token punctuation\">,</span> <span class=\"token string\">\"my_pref_value_column\"</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>\n</code></pre>\n<p>支持文件读取的 FileDataModel,Mahout 没有对文件的格式做过多的要求,只要文件的内容满足以下格式:</p>\n<ul>\n<li>每一行包括用户 ID, 物品 ID, 用户偏好</li>\n<li>逗号隔开或者 Tab 隔开</li>\n<li>*.zip 和 *.gz 文件会自动解压缩(Mahout 建议在数据量过大时采用压缩的数据存储)</li>\n</ul>\n<p>支持数据库读取的 JDBCDataModel,Mahout 提供一个默认的 MySQL 的支持,它对用户偏好数据的存放有以下简单的要求:</p>\n<ul>\n<li>用户 ID 列需要是 BIGINT 而且非空</li>\n<li>物品 ID 列需要是 BIGINT 而且非空</li>\n<li>用户偏好列需要是 FLOAT</li>\n</ul>\n<p>建议在用户 ID 和物品 ID 上建索引。</p>\n<h3><a id=\"Recommender_279\"></a>实现推荐:Recommender</h3>\n<p>介绍完数据表示模型,下面介绍 Mahout 提供的协同过滤的推荐策略,这里我们选择其中最经典的三种,User CF, Item CF 和 Slope One。</p>\n<p><strong>User CF</strong></p>\n<p>前面已经详细介绍了 User CF 的原理,这里我们着重看怎么基于 Mahout 实现 User CF 的推荐策略,我们还是从一个例子入手:</p>\n<h5><a id=\"_3__Mahout__User_CF_287\"></a>清单 3. 基于 Mahout 实现 User CF</h5>\n<pre><code class=\"prism language-java\">DataModel model <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">FileDataModel</span><span class=\"token punctuation\">(</span><span class=\"token keyword\">new</span> <span class=\"token class-name\">File</span><span class=\"token punctuation\">(</span><span class=\"token string\">\"preferences.dat\"</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nUserSimilarity similarity <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">PearsonCorrelationSimilarity</span><span class=\"token punctuation\">(</span>model<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nUserNeighborhood neighborhood <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">NearestNUserNeighborhood</span><span class=\"token punctuation\">(</span><span class=\"token number\">100</span><span class=\"token punctuation\">,</span> similarity<span class=\"token punctuation\">,</span> model<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nRecommender recommender <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">GenericUserBasedRecommender</span><span class=\"token punctuation\">(</span>model<span class=\"token punctuation\">,</span> \nneighborhood<span class=\"token punctuation\">,</span> similarity<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>\n</code></pre>\n<ol>\n<li>从文件建立 DataModel,我们采用前面介绍的 FileDataModel,这里假设用户的喜好信息存放在 preferences.dat 文件中。</li>\n<li>基于用户偏好数据计算用户的相似度,清单中采用的是 PearsonCorrelationSimilarity,前面章节曾详细介绍了各种计算相似度的方法,Mahout 中提供了基本的相似度的计算,它们都 UserSimilarity 这个接口,实现用户相似度的计算,包括下面这些常用的:</li>\n</ol>\n<ul>\n<li>PearsonCorrelationSimilarity:基于皮尔逊相关系数计算相似度</li>\n<li>EuclideanDistanceSimilarity:基于欧几里德距离计算相似度</li>\n<li>TanimotoCoefficientSimilarity:基于 Tanimoto 系数计算相似度</li>\n<li>UncerteredCosineSimilarity:计算 Cosine 相似度</li>\n</ul>\n<p>ItemSimilarity 也是类似的:</p>\n<ol>\n<li>根据建立的相似度计算方法,找到邻居用户。这里找邻居用户的方法根据前面我们介绍的,也包括两种:“固定数量的邻居”和“相似度门槛邻居”计算方法,Mahout 提供对应的实现:\n<ul>\n<li>NearestNUserNeighborhood:对每个用户取固定数量 N 的最近邻居</li>\n<li>ThresholdUserNeighborhood:对每个用户基于一定的限制,取落在相似度门限内的所有用户为邻居。</li>\n</ul>\n</li>\n<li>基于 DataModel,UserNeighborhood 和 UserSimilarity 构建 GenericUserBasedRecommender,实现 User CF 推荐策略。</li>\n</ol>\n<p><strong>Item CF</strong></p>\n<p>了解了 User CF,Mahout Item CF 的实现与 User CF 类似,是基于 ItemSimilarity,下面我们看实现的代码例子,它比 User CF 更简单,因为 Item CF 中并不需要引入邻居的概念:</p>\n<h5><a id=\"_4__Mahout__Item_CF_316\"></a>清单 4. 基于 Mahout 实现 Item CF</h5>\n<pre><code>`DataModel model = new FileDataModel(new File(\"preferences.dat\")); ``ItemSimilarity similarity = new PearsonCorrelationSimilarity(model); ``Recommender recommender = new GenericItemBasedRecommender(model, similarity);`\n</code></pre>\n<p><strong>Slope One</strong></p>\n<p>如前面介绍的,User CF 和 Item CF 是最常用最容易理解的两种 CF 的推荐策略,但在大数据量时,它们的计算量会很大,从而导致推荐效率较差。因此 Mahout 还提供了一种更加轻量级的 CF 推荐策略:Slope One。</p>\n<p>Slope One 是有 Daniel Lemire 和 Anna Maclachlan 在 2005 年提出的一种对基于评分的协同过滤推荐引擎的改进方法,下面简单介绍一下它的基本思想。</p>\n<p>图 5 给出了例子,假设系统对于物品 A,物品 B 和物品 C 的平均评分分别是 3,4 和 4。基于 Slope One 的方法会得到以下规律:</p>\n<ul>\n<li>用户对物品 B 的评分 = 用户对物品 A 的评分 + 1</li>\n<li>用户对物品 B 的评分 = 用户对物品 C 的评分</li>\n</ul>\n<p>基于以上的规律,我们可以对用户 A 和用户 B 的打分进行预测:</p>\n<ul>\n<li>对用户 A,他给物品 A 打分 4,那么我们可以推测他对物品 B 的评分是 5,对物品 C 的打分也是 5。</li>\n<li>对用户 B,他给物品 A 打分 2,给物品 C 打分 4,根据第一条规律,我们可以推断他对物品 B 的评分是 3;而根据第二条规律,推断出评分是 4。当出现冲突时,我们可以对各种规则得到的推断进行就平均,所以给出的推断是 3.5。</li>\n</ul>\n<p>这就是 Slope One 推荐的基本原理,它将用户的评分之间的关系看作简单的线性关系:</p>\n<p>Y = mX + b;</p>\n<p>当 m = 1 时就是 Slope One,也就是我们刚刚展示的例子。</p>\n<h5><a id=\"_5Slope_One__344\"></a>图 5.Slope One 推荐策略示例</h5>\n<p><img src=\"https://ws1.sinaimg.cn/large/006tNbRwly1fxev425is5g309i081aa6.gif\" alt=\"\"></p>\n<p>Slope One 的核心优势是在大规模的数据上,它依然能保证良好的计算速度和推荐效果。Mahout 提供了 Slope One 推荐方法的基本实现,实现代码很简单,参考清单 5.</p>\n<h5><a id=\"_5__Mahout__Slope_One_350\"></a>清单 5. 基于 Mahout 实现 Slope One</h5>\n<pre><code class=\"prism language-java\"><span class=\"token comment\">//In-Memory Recommender </span>\nDiffStorage diffStorage <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">MemoryDiffStorage</span><span class=\"token punctuation\">(</span>model<span class=\"token punctuation\">,</span> Weighting<span class=\"token punctuation\">.</span>UNWEIGHTED<span class=\"token punctuation\">,</span> <span class=\"token boolean\">false</span><span class=\"token punctuation\">,</span> \nLong<span class=\"token punctuation\">.</span>MAX_VALUE<span class=\"token punctuation\">)</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nRecommender recommender <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">SlopeOneRecommender</span><span class=\"token punctuation\">(</span>model<span class=\"token punctuation\">,</span> Weighting<span class=\"token punctuation\">.</span>UNWEIGHTED<span class=\"token punctuation\">,</span> \nWeighting<span class=\"token punctuation\">.</span>UNWEIGHTED<span class=\"token punctuation\">,</span> diffStorage<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \n \n<span class=\"token comment\">//Database-based Recommender </span>\nAbstractJDBCDataModel model <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">MySQLJDBCDataModel</span><span class=\"token punctuation\">(</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nDiffStorage diffStorage <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">MySQLJDBCDiffStorage</span><span class=\"token punctuation\">(</span>model<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span> \nRecommender recommender <span class=\"token operator\">=</span> <span class=\"token keyword\">new</span> <span class=\"token class-name\">SlopeOneRecommender</span><span class=\"token punctuation\">(</span>model<span class=\"token punctuation\">,</span> Weighting<span class=\"token punctuation\">.</span>WEIGHTED<span class=\"token punctuation\">,</span> \nWeighting<span class=\"token punctuation\">.</span>WEIGHTED<span class=\"token punctuation\">,</span> diffStorage<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>\n</code></pre>\n<ol>\n<li>\n<p>根据 Data Model 创建数据之间线性关系的模型 DiffStorage。</p>\n</li>\n<li>\n<p>基于 Data Model 和 DiffStorage 创建 SlopeOneRecommender,实现 Slope One 推荐策略。</p>\n</li>\n</ol>\n<h2><a id=\"_370\"></a>总结</h2>\n<p>Web2.0 的一个核心思想就是“集体智慧”,基于协同过滤的推荐策略的基本思想就是基于大众行为,为每个用户提供个性化的推荐,从而使用户能更快速更准确的发现所需要的信息。从应用角度分析,现今比较成功的推荐引擎,比如 Amazon,豆瓣,当当等都采用了协同过滤的方式,它不需要对物品或者用户进行严格的建模,而且不要求物品的描述是机器可理解的,是中领域无关的推荐方法,同时这个方法计算出来的推荐是开放的,可以共用他人的经验,很好的支持用户发现潜在的兴趣偏好。基于协同过滤的推荐策略也有不同的分支,它们有不同的实用场景和推荐效果,用户可以根据自己应用的实际情况选择合适的方法,异或组合不同的方法得到更好的推荐效果。</p>\n<p>除此之外,本文还介绍了如何基于 Apache Mahout 高效实现协同过滤推荐算法,Apache Mahout 关注海量数据上的机器学习经典算法的高效实现,其中对基于协同过滤的推荐方法也提供了很好的支持,基于 Mahout 你可以轻松的体验高效推荐的神奇。</p>\n<p>作为深入推荐引擎相关算法的第一篇文章,本文深入介绍了协同过滤算法,并举例介绍了如何基于 Apache Mahout 高效实现协同过滤推荐算法,Apache Mahout 作为海量数据上的机器学习经典算法的高效实现,其中对基于协同过滤的推荐方法也提供了很好的支持,基于 Mahout 你可以轻松的体验高效推荐的神奇。但我们也发现了在海量数据上高效的运行协同过滤算法以及其他推荐策略这样高复杂的算法还是有很大的挑战的。在面对这个问题的过程中,大家提出了很多减少计算量的方法,而聚类无疑是其中最优的选择。所以本系列的下一篇文章将详细介绍各类聚类算法,它们的原理,优缺点和实用场景,并给出基于 Apache Mahout 的聚类算法的高效实现,并分析在推荐引擎的实现中,如何通过引入聚类来解决大数据量造成的海量计算,从而提供高效的推荐。</p>\n<p>最后,感谢大家对本系列的关注和支持。</p>\n\n",
"markdowncontent":"# 深入推荐引擎相关算法 - 协同过滤\n\n## 集体智慧和协同过滤\n\n### 什么是集体智慧\n\n集体智慧 (Collective Intelligence) 并不是 Web2.0 时代特有的,只是在 Web2.0 时代,大家在 Web 应用中利用集体智慧构建更加有趣的应用或者得到更好的用户体验。集体智慧是指在大量的人群的行为和数据中收集答案,帮助你对整个人群得到统计意义上的结论,这些结论是我们在单个个体上无法得到的,它往往是某种趋势或者人群中共性的部分。\n\nWikipedia 和 Google 是两个典型的利用集体智慧的 Web 2.0 应用:\n\n- Wikipedia 是一个知识管理的百科全书,相对于传统的由领域专家编辑的百科全书,Wikipedia 允许最终用户贡献知识,随着参与人数的增多,Wikipedia 变成了涵盖各个领域的一本无比全面的知识库。也许有人会质疑它的权威性,但如果你从另一个侧面想这个问题,也许就可以迎刃而解。在发行一本书时,作者虽然是权威,但难免还有一些错误,然后通过一版一版的改版,书的内容越来越完善。而在 Wikipedia 上,这种改版和修正被变为每个人都可以做的事情,任何人发现错误或者不完善都可以贡献他们的想法,即便某些信息是错误的,但它一定也会尽快的被其他人纠正过来。从一个宏观的角度看,整个系统在按照一个良性循环的轨迹不断完善,这也正是集体智慧的魅力。\n- Google:目前最流行的搜索引擎,与 Wikipedia 不同,它没有要求用户显式的贡献,但仔细想想 Google 最核心的 PageRank 的思想,它利用了 Web 页面之间的关系,将多少其他页面链接到当前页面的数目作为衡量当前页面重要与否的标准;如果这不好理解,那么你可以把它想象成一个选举的过程,每个 Web 页面都是一个投票者同时也是一个被投票者,PageRank 通过一定数目的迭代得到一个相对稳定的评分。Google 其实利用了现在 Internet 上所有 Web 页面上链接的集体智慧,找到哪些页面是重要的。\n\n### 什么是协同过滤\n\n协同过滤是利用集体智慧的一个典型方法。要理解什么是协同过滤 (Collaborative Filtering, 简称 CF),首先想一个简单的问题,如果你现在想看个电影,但你不知道具体看哪部,你会怎么做?大部分的人会问问周围的朋友,看看最近有什么好看的电影推荐,而我们一般更倾向于从口味比较类似的朋友那里得到推荐。这就是协同过滤的核心思想。\n\n协同过滤一般是在海量的用户中发掘出一小部分和你品位比较类似的,在协同过滤中,这些用户成为邻居,然后根据他们喜欢的其他东西组织成一个排序的目录作为推荐给你。当然其中有一个核心的问题:\n\n- 如何确定一个用户是不是和你有相似的品位?\n- 如何将邻居们的喜好组织成一个排序的目录?\n\n协同过滤相对于集体智慧而言,它从一定程度上保留了个体的特征,就是你的品位偏好,所以它更多可以作为个性化推荐的算法思想。可以想象,这种推荐策略在 Web 2.0 的长尾中是很重要的,将大众流行的东西推荐给长尾中的人怎么可能得到好的效果,这也回到推荐系统的一个核心问题:了解你的用户,然后才能给出更好的推荐。\n\n### 深入协同过滤的核心\n\n前面作为背景知识,介绍了集体智慧和协同过滤的基本思想,这一节我们将深入分析协同过滤的原理,介绍基于协同过滤思想的多种推荐机制,优缺点和实用场景。\n\n首先,要实现协同过滤,需要一下几个步骤\n\n- 收集用户偏好\n- 找到相似的用户或物品\n- 计算推荐\n\n### 收集用户偏好\n\n要从用户的行为和偏好中发现规律,并基于此给予推荐,如何收集用户的偏好信息成为系统推荐效果最基础的决定因素。用户有很多方式向系统提供自己的偏好信息,而且不同的应用也可能大不相同,下面举例进行介绍:\n\n##### 表 1 用户行为和用户偏好\n\n| **用户行为** | **类型** | **特征** | **作用** |\n| ---------------- | -------- | ------------------------------------------------------------ | ------------------------------------------------------------ |\n| 评分 | 显式 | 整数量化的偏好,可能的取值是 [0, n];n 一般取值为 5 或者是 10 | 通过用户对物品的评分,可以精确的得到用户的偏好 |\n| 投票 | 显式 | 布尔量化的偏好,取值是 0 或 1 | 通过用户对物品的投票,可以较精确的得到用户的偏好 |\n| 转发 | 显式 | 布尔量化的偏好,取值是 0 或 1 | 通过用户对物品的投票,可以精确的得到用户的偏好。 如果是站内,同时可以推理得到被转发人的偏好(不精确) |\n| 保存书签 | 显示 | 布尔量化的偏好,取值是 0 或 1 | 通过用户对物品的投票,可以精确的得到用户的偏好。 |\n| 标记标签 (Tag) | 显示 | 一些单词,需要对单词进行分析,得到偏好 | 通过分析用户的标签,可以得到用户对项目的理解,同时可以分析出用户的情感:喜欢还是讨厌 |\n| 评论 | 显示 | 一段文字,需要进行文本分析,得到偏好 | 通过分析用户的评论,可以得到用户的情感:喜欢还是讨厌 |\n| 点击流 ( 查看 ) | 隐式 | 一组用户的点击,用户对物品感兴趣,需要进行分析,得到偏好 | 用户的点击一定程度上反映了用户的注意力,所以它也可以从一定程度上反映用户的喜好。 |\n| 页面停留时间 | 隐式 | 一组时间信息,噪音大,需要进行去噪,分析,得到偏好 | 用户的页面停留时间一定程度上反映了用户的注意力和喜好,但噪音偏大,不好利用。 |\n| 购买 | 隐式 | 布尔量化的偏好,取值是 0 或 1 | 用户的购买是很明确的说明这个项目它感兴趣。 |\n\n以上列举的用户行为都是比较通用的,推荐引擎设计人员可以根据自己应用的特点添加特殊的用户行为,并用他们表示用户对物品的喜好。\n\n在一般应用中,我们提取的用户行为一般都多于一种,关于如何组合这些不同的用户行为,基本上有以下两种方式:\n\n- 将不同的行为分组:一般可以分为“查看”和“购买”等等,然后基于不同的行为,计算不同的用户 / 物品相似度。类似于当当网或者 Amazon 给出的“购买了该图书的人还购买了 ...”,“查看了图书的人还查看了 ...”\n- 根据不同行为反映用户喜好的程度将它们进行加权,得到用户对于物品的总体喜好。一般来说,显式的用户反馈比隐式的权值大,但比较稀疏,毕竟进行显示反馈的用户是少数;同时相对于“查看”,“购买”行为反映用户喜好的程度更大,但这也因应用而异。\n\n收集了用户行为数据,我们还需要对数据进行一定的预处理,其中最核心的工作就是:减噪和归一化。\n\n- 减噪:用户行为数据是用户在使用应用过程中产生的,它可能存在大量的噪音和用户的误操作,我们可以通过经典的数据挖掘算法过滤掉行为数据中的噪音,这样可以是我们的分析更加精确。\n- 归一化:如前面讲到的,在计算用户对物品的喜好程度时,可能需要对不同的行为数据进行加权。但可以想象,不同行为的数据取值可能相差很大,比如,用户的查看数据必然比购买数据大的多,如何将各个行为的数据统一在一个相同的取值范围中,从而使得加权求和得到的总体喜好更加精确,就需要我们进行归一化处理。最简单的归一化处理,就是将各类数据除以此类中的最大值,以保证归一化后的数据取值在 [0,1] 范围中。\n\n进行的预处理后,根据不同应用的行为分析方法,可以选择分组或者加权处理,之后我们可以得到一个用户偏好的二维矩阵,一维是用户列表,另一维是物品列表,值是用户对物品的偏好,一般是 [0,1] 或者 [-1, 1] 的浮点数值。\n\n### 找到相似的用户或物品\n\n当已经对用户行为进行分析得到用户喜好后,我们可以根据用户喜好计算相似用户和物品,然后基于相似用户或者物品进行推荐,这就是最典型的 CF 的两个分支:基于用户的 CF 和基于物品的 CF。这两种方法都需要计算相似度,下面我们先看看最基本的几种计算相似度的方法。\n\n**相似度的计算**\n\n关于相似度的计算,现有的几种基本方法都是基于向量(Vector)的,其实也就是计算两个向量的距离,距离越近相似度越大。在推荐的场景中,在用户 - 物品偏好的二维矩阵中,我们可以将一个用户对所有物品的偏好作为一个向量来计算用户之间的相似度,或者将所有用户对某个物品的偏好作为一个向量来计算物品之间的相似度。下面我们详细介绍几种常用的相似度计算方法:\n\n- 欧几里德距离(Euclidean Distance)\n\n最初用于计算欧几里德空间中两个点的距离,假设 x,y 是 n 维空间的两个点,它们之间的欧几里德距离是:\n\n![Figure xxx. Requires a heading](https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image003.gif)\n\n可以看出,当 n=2 时,欧几里德距离就是平面上两个点的距离。欧几里德学习视频: [《机器学习Python实践》#15 欧几里得距离](https://www.yxgapp.com/video/c8426884-2b56-494f-a274-0aa3105503f1.html)\n\n当用欧几里德距离表示相似度,一般采用以下公式进行转换:距离越小,相似度越大\n\n![Figure xxx. Requires a heading](https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image005.gif)\n\n- 皮尔逊相关系数(Pearson Correlation Coefficient)\n\n皮尔逊相关系数一般用于计算两个定距变量间联系的紧密程度,它的取值在 [-1,+1] 之间。\n\n![Figure xxx. Requires a heading](https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image007.gif)\n\n\n\nsx, sy是 x 和 y 的样品标准偏差。\n\n- Cosine 相似度(Cosine Similarity)\n\nCosine 相似度被广泛应用于计算文档数据的相似度:\n\n![Figure xxx. Requires a heading](https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image009.gif)\n\n\n\n\n\n- Tanimoto 系数(Tanimoto Coefficient)\n\nTanimoto 系数也称为 Jaccard 系数,是 Cosine 相似度的扩展,也多用于计算文档数据的相似度:\n\n![Figure xxx. Requires a heading](https://www.ibm.com/developerworks/cn/web/1103_zhaoct_recommstudy2/image011.gif)\n\n**相似邻居的计算**\n\n介绍完相似度的计算方法,下面我们看看如何根据相似度找到用户 - 物品的邻居,常用的挑选邻居的原则可以分为两类:图 1 给出了二维平面空间上点集的示意图。\n\n- 固定数量的邻居:K-neighborhoods 或者 Fix-size neighborhoods\n\n不论邻居的“远近”,只取最近的 K 个,作为其邻居。如图 1 中的 A,假设要计算点 1 的 5- 邻居,那么根据点之间的距离,我们取最近的 5 个点,分别是点 2,点 3,点 4,点 7 和点 5。但很明显我们可以看出,这种方法对于孤立点的计算效果不好,因为要取固定个数的邻居,当它附近没有足够多比较相似的点,就被迫取一些不太相似的点作为邻居,这样就影响了邻居相似的程度,比如图 1 中,点 1 和点 5 其实并不是很相似。\n\n- 基于相似度门槛的邻居:Threshold-based neighborhoods\n\n与计算固定数量的邻居的原则不同,基于相似度门槛的邻居计算是对邻居的远近进行最大值的限制,落在以当前点为中心,距离为 K 的区域中的所有点都作为当前点的邻居,这种方法计算得到的邻居个数不确定,但相似度不会出现较大的误差。如图 1 中的 B,从点 1 出发,计算相似度在 K 内的邻居,得到点 2,点 3,点 4 和点 7,这种方法计算出的邻居的相似度程度比前一种优,尤其是对孤立点的处理。\n\n##### 图 1.相似邻居计算示意图\n\n![](https://ws4.sinaimg.cn/large/006tNbRwly1fxev00vhkxg30ej08iwef.gif)\n\n### 计算推荐\n\n经过前期的计算已经得到了相邻用户和相邻物品,下面介绍如何基于这些信息为用户进行推荐。本系列的上一篇综述文章已经简要介绍过基于协同过滤的推荐算法可以分为基于用户的 CF 和基于物品的 CF,下面我们深入这两种方法的计算方法,使用场景和优缺点。\n\n**基于用户的 CF(User CF)**\n\n基于用户的 CF 的基本思想相当简单,基于用户对物品的偏好找到相邻邻居用户,然后将邻居用户喜欢的推荐给当前用户。计算上,就是将一个用户对所有物品的偏好作为一个向量来计算用户之间的相似度,找到 K 邻居后,根据邻居的相似度权重以及他们对物品的偏好,预测当前用户没有偏好的未涉及物品,计算得到一个排序的物品列表作为推荐。图 2 给出了一个例子,对于用户 A,根据用户的历史偏好,这里只计算得到一个邻居 - 用户 C,然后将用户 C 喜欢的物品 D 推荐给用户 A。\n\n##### 图 2.基于用户的 CF 的基本原理\n\n![](https://ws3.sinaimg.cn/large/006tNbRwly1fxev0k5q29g30aw08wglm.gif)\n\n**基于物品的 CF(Item CF)**\n\n基于物品的 CF 的原理和基于用户的 CF 类似,只是在计算邻居时采用物品本身,而不是从用户的角度,即基于用户对物品的偏好找到相似的物品,然后根据用户的历史偏好,推荐相似的物品给他。从计算的角度看,就是将所有用户对某个物品的偏好作为一个向量来计算物品之间的相似度,得到物品的相似物品后,根据用户历史的偏好预测当前用户还没有表示偏好的物品,计算得到一个排序的物品列表作为推荐。图 3 给出了一个例子,对于物品 A,根据所有用户的历史偏好,喜欢物品 A 的用户都喜欢物品 C,得出物品 A 和物品 C 比较相似,而用户 C 喜欢物品 A,那么可以推断出用户 C 可能也喜欢物品 C。\n\n##### 图 3.基于物品的 CF 的基本原理\n\n![\u0086](https://ws2.sinaimg.cn/large/006tNbRwly1fxev0tb00rg30aa099glm.gif)\n\n**User CF vs. Item CF**\n\n前面介绍了 User CF 和 Item CF 的基本原理,下面我们分几个不同的角度深入看看它们各自的优缺点和适用场景:\n\n- 计算复杂度\n\nItem CF 和 User CF 是基于协同过滤推荐的两个最基本的算法,User CF 是很早以前就提出来了,Item CF 是从 Amazon 的论文和专利发表之后(2001 年左右)开始流行,大家都觉得 Item CF 从性能和复杂度上比 User CF 更优,其中的一个主要原因就是对于一个在线网站,用户的数量往往大大超过物品的数量,同时物品的数据相对稳定,因此计算物品的相似度不但计算量较小,同时也不必频繁更新。但我们往往忽略了这种情况只适应于提供商品的电子商务网站,对于新闻,博客或者微内容的推荐系统,情况往往是相反的,物品的数量是海量的,同时也是更新频繁的,所以单从复杂度的角度,这两个算法在不同的系统中各有优势,推荐引擎的设计者需要根据自己应用的特点选择更加合适的算法。\n\n- 适用场景\n\n在非社交网络的网站中,内容内在的联系是很重要的推荐原则,它比基于相似用户的推荐原则更加有效。比如在购书网站上,当你看一本书的时候,推荐引擎会给你推荐相关的书籍,这个推荐的重要性远远超过了网站首页对该用户的综合推荐。可以看到,在这种情况下,Item CF 的推荐成为了引导用户浏览的重要手段。同时 Item CF 便于为推荐做出解释,在一个非社交网络的网站中,给某个用户推荐一本书,同时给出的解释是某某和你有相似兴趣的人也看了这本书,这很难让用户信服,因为用户可能根本不认识那个人;但如果解释说是因为这本书和你以前看的某本书相似,用户可能就觉得合理而采纳了此推荐。\n\n相反的,在现今很流行的社交网络站点中,User CF 是一个更不错的选择,User CF 加上社会网络信息,可以增加用户对推荐解释的信服程度。\n\n- 推荐多样性和精度\n\n研究推荐引擎的学者们在相同的数据集合上分别用 User CF 和 Item CF 计算推荐结果,发现推荐列表中,只有 50% 是一样的,还有 50% 完全不同。但是这两个算法确有相似的精度,所以可以说,这两个算法是很互补的。\n\n关于推荐的多样性,有两种度量方法:\n\n第一种度量方法是从单个用户的角度度量,就是说给定一个用户,查看系统给出的推荐列表是否多样,也就是要比较推荐列表中的物品之间两两的相似度,不难想到,对这种度量方法,Item CF 的多样性显然不如 User CF 的好,因为 Item CF 的推荐就是和以前看的东西最相似的。\n\n第二种度量方法是考虑系统的多样性,也被称为覆盖率 (Coverage),它是指一个推荐系统是否能够提供给所有用户丰富的选择。在这种指标下,Item CF 的多样性要远远好于 User CF, 因为 User CF 总是倾向于推荐热门的,从另一个侧面看,也就是说,Item CF 的推荐有很好的新颖性,很擅长推荐长尾里的物品。所以,尽管大多数情况,Item CF 的精度略小于 User CF, 但如果考虑多样性,Item CF 却比 User CF 好很多。\n\n如果你对推荐的多样性还心存疑惑,那么下面我们再举个实例看看 User CF 和 Item CF 的多样性到底有什么差别。首先,假设每个用户兴趣爱好都是广泛的,喜欢好几个领域的东西,不过每个用户肯定也有一个主要的领域,对这个领域会比其他领域更加关心。给定一个用户,假设他喜欢 3 个领域 A,B,C,A 是他喜欢的主要领域,这个时候我们来看 User CF 和 Item CF 倾向于做出什么推荐:如果用 User CF, 它会将 A,B,C 三个领域中比较热门的东西推荐给用户;而如果用 ItemCF,它会基本上只推荐 A 领域的东西给用户。所以我们看到因为 User CF 只推荐热门的,所以它在推荐长尾里项目方面的能力不足;而 Item CF 只推荐 A 领域给用户,这样他有限的推荐列表中就可能包含了一定数量的不热门的长尾物品,同时 Item CF 的推荐对这个用户而言,显然多样性不足。但是对整个系统而言,因为不同的用户的主要兴趣点不同,所以系统的覆盖率会比较好。\n\n从上面的分析,可以很清晰的看到,这两种推荐都有其合理性,但都不是最好的选择,因此他们的精度也会有损失。其实对这类系统的最好选择是,如果系统给这个用户推荐 30 个物品,既不是每个领域挑选 10 个最热门的给他,也不是推荐 30 个 A 领域的给他,而是比如推荐 15 个 A 领域的给他,剩下的 15 个从 B,C 中选择。所以结合 User CF 和 Item CF 是最优的选择,结合的基本原则就是当采用 Item CF 导致系统对个人推荐的多样性不足时,我们通过加入 User CF 增加个人推荐的多样性,从而提高精度,而当因为采用 User CF 而使系统的整体多样性不足时,我们可以通过加入 Item CF 增加整体的多样性,同样同样可以提高推荐的精度。\n\n- 用户对推荐算法的适应度\n\n前面我们大部分都是从推荐引擎的角度考虑哪个算法更优,但其实我们更多的应该考虑作为推荐引擎的最终使用者 -- 应用用户对推荐算法的适应度。\n\n对于 User CF,推荐的原则是假设用户会喜欢那些和他有相同喜好的用户喜欢的东西,但如果一个用户没有相同喜好的朋友,那 User CF 的算法的效果就会很差,所以一个用户对的 CF 算法的适应度是和他有多少共同喜好用户成正比的。\n\nItem CF 算法也有一个基本假设,就是用户会喜欢和他以前喜欢的东西相似的东西,那么我们可以计算一个用户喜欢的物品的自相似度。一个用户喜欢物品的自相似度大,就说明他喜欢的东西都是比较相似的,也就是说他比较符合 Item CF 方法的基本假设,那么他对 Item CF 的适应度自然比较好;反之,如果自相似度小,就说明这个用户的喜好习惯并不满足 Item CF 方法的基本假设,那么对于这种用户,用 Item CF 方法做出好的推荐的可能性非常低。\n\n通过以上的介绍,相信大家已经对协同过滤推荐的各种方法,原则,特点和适用场景有深入的了解,下面我们就进入实战阶段,重点介绍如何基于 Apache Mahout 实现协同过滤推荐算法。\n\n## 基于 Apache Mahout 实现高效的协同过滤推荐\n\nApache Mahout 是 Apache Software Foundation (ASF) 旗下的一个开源项目,提供一些可扩展的机器学习领域经典算法的实现,旨在帮助开发人员更加方便快捷地创建智能应用程序,并且,在 Mahout 的最近版本中还加入了对 Apache Hadoop 的支持,使这些算法可以更高效的运行在云计算环境中。\n\n关于 Apache Mahout 的安装和配置请参考《基于 Apache Mahout 构建社会化推荐引擎》,它是笔者 09 年发表的一篇关于基于 Mahout 实现推荐引擎的 developerWorks 文章,其中详细介绍了 Mahout 的安装步骤,并给出一个简单的电影推荐引擎的例子。\n\nApache Mahout 中提供的一个协同过滤算法的高效实现,它是一个基于 Java 实现的可扩展的,高效的推荐引擎。图 4 给出了 Apache Mahout 中协同过滤推荐实现的组件图,下面我们逐步深入介绍各个部分。\n\n##### 图 4.组件图\n\n![](https://ws3.sinaimg.cn/large/006tNbRwly1fxev2pr7jeg30cs09zglj.gif)\n\n### 数据表示:Data Model\n\n**Preference**\n\n基于协同过滤的推荐引擎的输入是用户的历史偏好信息,在 Mahout 里它被建模为 Preference(接口),一个 Preference 就是一个简单的三元组 < 用户 ID, 物品 ID, 用户偏好 >,它的实现类是 GenericPreference,可以通过以下语句创建一个 GenericPreference。\n\nGenericPreference preference = new GenericPreference(123, 456, 3.0f);\n\n这其中, 123 是用户 ID,long 型;456 是物品 ID,long 型;3.0f 是用户偏好,float 型。从这个例子我们可以看出,单单一个 GenericPreference 的数据就占用 20 bytes,所以你会发现如果只简单实用数组 Array 加载用户偏好数据,必然占用大量的内存,Mahout 在这方面做了一些优化,它创建了 PreferenceArray(接口)保存一组用户偏好数据,为了优化性能,Mahout 给出了两个实现类,GenericUserPreferenceArray 和 GenericItemPreferenceArray,分别按照用户和物品本身对用户偏好进行组装,这样就可以压缩用户 ID 或者物品 ID 的空间。下面清单 1 的代码以 GenericUserPreferenceArray 为例,展示了如何创建和使用一个 PreferenceArray。\n\n##### 清单 1. 创建和使用 PreferenceArray\n\n```java\nPreferenceArray userPref = new GenericUserPreferenceArray(2); //size = 2 \n \nuserPref.setUserID(0, 1L); \n \nuserPref.setItemID(0, 101L); //<1L, 101L, 2.0f> \nuserPref.setValue(0, 2.0f); \nuserPref.setItemID(1, 102L); //<1L, 102L, 4.0f> \nuserPref.setValue(1, 4.0f); \n \nPreference pref = userPref.get(1); //<1L, 102L, 4.0f>\n```\n\n为了提高性能 Mahout 还构建了自己的 HashMap 和 Set:FastByIDMap 和 FastIDSet,有兴趣的朋友可以参考 Mahout 官方说明。\n\n**DataModel**\n\nMahout 的推荐引擎实际接受的输入是 DataModel,它是对用户偏好数据的压缩表示,通过创建内存版 DataModel 的语句我们可以看出:\n\nDataModel model = new GenericDataModel(FastByIDMap<PreferenceArray> map);\n\n他保存在一个按照用户 ID 或者物品 ID 进行散列的 PreferenceArray,而 PreferenceArray 中对应保存着这个用户 ID 或者物品 ID 的所有用户偏好信息。\n\nDataModel 是用户喜好信息的抽象接口,它的具体实现支持从任意类型的数据源抽取用户喜好信息,具体实现包括内存版的 GenericDataModel,支持文件读取的 FileDataModel 和支持数据库读取的 JDBCDataModel,下面我们看看如何创建各种 DataModel。\n\n##### 清单 2. 创建各种 DataModel\n\n```java\n//In-memory DataModel - GenericDataModel \n FastByIDMap<PreferenceArray> preferences = new FastByIDMap<PreferenceArray>(); \n \n PreferenceArray prefsForUser1 = new GenericUserPreferenceArray(10); \n prefsForUser1.setUserID(0, 1L); \n prefsForUser1.setItemID(0, 101L); \n prefsForUser1.setValue(0, 3.0f); \n prefsForUser1.setItemID(1, 102L); \n prefsForUser1.setValue(1, 4.5f); \n… (8 more) \n preferences.put(1L, prefsForUser1); //use userID as the key \n… (more users) \n \n DataModel model = new GenericDataModel(preferences); \n \n //File-based DataModel - FileDataModel \n DataModel dataModel = new FileDataModel(new File(\"preferences.csv\"); \n \n //Database-based DataModel - MySQLJDBCDataModel \n MysqlDataSource dataSource = new MysqlDataSource(); \n dataSource.setServerName(\"my_user\"); \n dataSource.setUser(\"my_password\"); \n dataSource.setPassword(\"my_database_host\"); \n JDBCDataModel dataModel = new MySQLJDBCDataModel(dataSource, \"my_prefs_table\", \n \"my_user_column\", \"my_item_column\", \"my_pref_value_column\");\n```\n\n支持文件读取的 FileDataModel,Mahout 没有对文件的格式做过多的要求,只要文件的内容满足以下格式:\n\n- 每一行包括用户 ID, 物品 ID, 用户偏好\n- 逗号隔开或者 Tab 隔开\n- *.zip 和 *.gz 文件会自动解压缩(Mahout 建议在数据量过大时采用压缩的数据存储)\n\n支持数据库读取的 JDBCDataModel,Mahout 提供一个默认的 MySQL 的支持,它对用户偏好数据的存放有以下简单的要求:\n\n- 用户 ID 列需要是 BIGINT 而且非空\n- 物品 ID 列需要是 BIGINT 而且非空\n- 用户偏好列需要是 FLOAT\n\n建议在用户 ID 和物品 ID 上建索引。\n\n### 实现推荐:Recommender\n\n介绍完数据表示模型,下面介绍 Mahout 提供的协同过滤的推荐策略,这里我们选择其中最经典的三种,User CF, Item CF 和 Slope One。\n\n**User CF**\n\n前面已经详细介绍了 User CF 的原理,这里我们着重看怎么基于 Mahout 实现 User CF 的推荐策略,我们还是从一个例子入手:\n\n##### 清单 3. 基于 Mahout 实现 User CF\n\n```java\nDataModel model = new FileDataModel(new File(\"preferences.dat\")); \nUserSimilarity similarity = new PearsonCorrelationSimilarity(model); \nUserNeighborhood neighborhood = new NearestNUserNeighborhood(100, similarity, model); \nRecommender recommender = new GenericUserBasedRecommender(model, \nneighborhood, similarity);\n```\n\n1. 从文件建立 DataModel,我们采用前面介绍的 FileDataModel,这里假设用户的喜好信息存放在 preferences.dat 文件中。\n2. 基于用户偏好数据计算用户的相似度,清单中采用的是 PearsonCorrelationSimilarity,前面章节曾详细介绍了各种计算相似度的方法,Mahout 中提供了基本的相似度的计算,它们都 UserSimilarity 这个接口,实现用户相似度的计算,包括下面这些常用的:\n\n- PearsonCorrelationSimilarity:基于皮尔逊相关系数计算相似度\n- EuclideanDistanceSimilarity:基于欧几里德距离计算相似度\n- TanimotoCoefficientSimilarity:基于 Tanimoto 系数计算相似度\n- UncerteredCosineSimilarity:计算 Cosine 相似度\n\nItemSimilarity 也是类似的:\n\n1. 根据建立的相似度计算方法,找到邻居用户。这里找邻居用户的方法根据前面我们介绍的,也包括两种:“固定数量的邻居”和“相似度门槛邻居”计算方法,Mahout 提供对应的实现:\n - NearestNUserNeighborhood:对每个用户取固定数量 N 的最近邻居\n - ThresholdUserNeighborhood:对每个用户基于一定的限制,取落在相似度门限内的所有用户为邻居。\n2. 基于 DataModel,UserNeighborhood 和 UserSimilarity 构建 GenericUserBasedRecommender,实现 User CF 推荐策略。\n\n**Item CF**\n\n了解了 User CF,Mahout Item CF 的实现与 User CF 类似,是基于 ItemSimilarity,下面我们看实现的代码例子,它比 User CF 更简单,因为 Item CF 中并不需要引入邻居的概念:\n\n##### 清单 4. 基于 Mahout 实现 Item CF\n\n```\n`DataModel model = new FileDataModel(new File(\"preferences.dat\")); ``ItemSimilarity similarity = new PearsonCorrelationSimilarity(model); ``Recommender recommender = new GenericItemBasedRecommender(model, similarity);`\n```\n\n**Slope One**\n\n如前面介绍的,User CF 和 Item CF 是最常用最容易理解的两种 CF 的推荐策略,但在大数据量时,它们的计算量会很大,从而导致推荐效率较差。因此 Mahout 还提供了一种更加轻量级的 CF 推荐策略:Slope One。\n\nSlope One 是有 Daniel Lemire 和 Anna Maclachlan 在 2005 年提出的一种对基于评分的协同过滤推荐引擎的改进方法,下面简单介绍一下它的基本思想。\n\n图 5 给出了例子,假设系统对于物品 A,物品 B 和物品 C 的平均评分分别是 3,4 和 4。基于 Slope One 的方法会得到以下规律:\n\n- 用户对物品 B 的评分 = 用户对物品 A 的评分 + 1\n- 用户对物品 B 的评分 = 用户对物品 C 的评分\n\n基于以上的规律,我们可以对用户 A 和用户 B 的打分进行预测:\n\n- 对用户 A,他给物品 A 打分 4,那么我们可以推测他对物品 B 的评分是 5,对物品 C 的打分也是 5。\n- 对用户 B,他给物品 A 打分 2,给物品 C 打分 4,根据第一条规律,我们可以推断他对物品 B 的评分是 3;而根据第二条规律,推断出评分是 4。当出现冲突时,我们可以对各种规则得到的推断进行就平均,所以给出的推断是 3.5。\n\n这就是 Slope One 推荐的基本原理,它将用户的评分之间的关系看作简单的线性关系:\n\nY = mX + b;\n\n当 m = 1 时就是 Slope One,也就是我们刚刚展示的例子。\n\n##### 图 5.Slope One 推荐策略示例\n\n![](https://ws1.sinaimg.cn/large/006tNbRwly1fxev425is5g309i081aa6.gif)\n\nSlope One 的核心优势是在大规模的数据上,它依然能保证良好的计算速度和推荐效果。Mahout 提供了 Slope One 推荐方法的基本实现,实现代码很简单,参考清单 5.\n\n##### 清单 5. 基于 Mahout 实现 Slope One\n\n```java\n//In-Memory Recommender \nDiffStorage diffStorage = new MemoryDiffStorage(model, Weighting.UNWEIGHTED, false, \nLong.MAX_VALUE)); \nRecommender recommender = new SlopeOneRecommender(model, Weighting.UNWEIGHTED, \nWeighting.UNWEIGHTED, diffStorage); \n \n//Database-based Recommender \nAbstractJDBCDataModel model = new MySQLJDBCDataModel(); \nDiffStorage diffStorage = new MySQLJDBCDiffStorage(model); \nRecommender recommender = new SlopeOneRecommender(model, Weighting.WEIGHTED, \nWeighting.WEIGHTED, diffStorage);\n```\n\n1. 根据 Data Model 创建数据之间线性关系的模型 DiffStorage。\n\n2. 基于 Data Model 和 DiffStorage 创建 SlopeOneRecommender,实现 Slope One 推荐策略。\n\n## 总结\n\nWeb2.0 的一个核心思想就是“集体智慧”,基于协同过滤的推荐策略的基本思想就是基于大众行为,为每个用户提供个性化的推荐,从而使用户能更快速更准确的发现所需要的信息。从应用角度分析,现今比较成功的推荐引擎,比如 Amazon,豆瓣,当当等都采用了协同过滤的方式,它不需要对物品或者用户进行严格的建模,而且不要求物品的描述是机器可理解的,是中领域无关的推荐方法,同时这个方法计算出来的推荐是开放的,可以共用他人的经验,很好的支持用户发现潜在的兴趣偏好。基于协同过滤的推荐策略也有不同的分支,它们有不同的实用场景和推荐效果,用户可以根据自己应用的实际情况选择合适的方法,异或组合不同的方法得到更好的推荐效果。\n\n除此之外,本文还介绍了如何基于 Apache Mahout 高效实现协同过滤推荐算法,Apache Mahout 关注海量数据上的机器学习经典算法的高效实现,其中对基于协同过滤的推荐方法也提供了很好的支持,基于 Mahout 你可以轻松的体验高效推荐的神奇。\n\n作为深入推荐引擎相关算法的第一篇文章,本文深入介绍了协同过滤算法,并举例介绍了如何基于 Apache Mahout 高效实现协同过滤推荐算法,Apache Mahout 作为海量数据上的机器学习经典算法的高效实现,其中对基于协同过滤的推荐方法也提供了很好的支持,基于 Mahout 你可以轻松的体验高效推荐的神奇。但我们也发现了在海量数据上高效的运行协同过滤算法以及其他推荐策略这样高复杂的算法还是有很大的挑战的。在面对这个问题的过程中,大家提出了很多减少计算量的方法,而聚类无疑是其中最优的选择。所以本系列的下一篇文章将详细介绍各类聚类算法,它们的原理,优缺点和实用场景,并给出基于 Apache Mahout 的聚类算法的高效实现,并分析在推荐引擎的实现中,如何通过引入聚类来解决大数据量造成的海量计算,从而提供高效的推荐。\n\n最后,感谢大家对本系列的关注和支持。\n",
"tags":"协同过滤,个性化推荐",
"id":"84310535",
"categories":"java",
"status":1
},
"error":"",
"status":true
}
]
- 转换为hexo格式的md文档
java -jar csdn-blog-export-0.0.1-SNAPSHOT.jar g --genDir=生成路径
生成路径为一个文件夹,跳转到指定文件夹即可。