ALS音乐推荐(上)

  本篇文章的开头笔者提出一个疑问,何为数据科学,数据科学是作什么的?你们带着这个疑问去读接下来的这篇音乐推荐的公众号。html

  从经验上讲,推荐引擎属于大规模机器学习,在平常购物中你们或许深有体会,好比:你在淘宝上浏览了一些商品,或者购买了一些商品,那么淘宝就会根据你的偏好给你推荐一些其余相似的商品。然而,相比较其余机器学习算法,推荐引擎的输出更加的直观,有时候的推荐效果让人吃惊。做为机器学习开篇文章,本篇文章会系统的介绍基于Audioscrobbler数据集的音乐推荐。ios

数据集介绍算法

  Audioscrobbler数据集是一个公开发布的数据集,读者能够在(http://www-etud.iro.umontreal.ca/~bergstj/audioscrobbler_data.html)网站获取。数据集主要有三部分组成,user_artist_data.txt文件是主要的数据集文件记录了约2420条用户id、艺术家id以及用户收听艺术家歌曲的次数数据,包含141000个用户和160万个艺术家;artist_data.txt文件记录了艺术家id和对应的名字;artist_alias.txt记录了艺术家id和对应的别称id。微信

推荐算法介绍dom

  因为所选取的数据集只记录了用户和歌曲之间的交互状况,除了艺术家名字以外没有其余信息。所以要找的学习算法不须要用户和艺术家的属性信息,这类算法一般被称为协同过滤。若是根据两个用户的年龄相同来判断他们可能具备类似的偏好,这不叫协同过滤。相反,根据两个用户播放过许多相同歌曲来判断他们可能都喜欢某首歌,这是协调过滤。机器学习

  本篇所用的算法在数学上称为迭代最小二乘,把用户播放数据当成矩阵A,矩阵低i行第j列上的元素的值,表明用户i播放艺术家j的音乐。矩阵A是稀疏的,绝大多数元素是0,算法将A分解成两个小矩阵X和Y,既A=XYT,X表明用户特征矩阵,Y表明特征艺术家矩阵。两个矩阵的乘积当作用户-艺术家关系矩阵的估计。能够经过下边一组图直观的反映:学习

  如今假若有5个听众,音乐有5首,那么A是一个5*5的矩阵,假如评分以下:大数据

图2.1 用户订阅矩阵优化

  假如d是三个属性,那么X的矩阵以下:网站

图2.2 用户-特征矩阵

  Y的矩阵以下:

图2.3 特征-电影矩阵

  实际的求解过程当中一般先随机的固定矩阵Y,则,为提升计算效率,一般采用并行计算X的每一行,既。获得X以后,再反求出Y,不断的交替迭代,最终使得XYT与A的平方偏差小于指定阈值,中止迭代,获得最终的X(表明用户特征矩阵)和Y矩阵(表明特征艺术家矩阵)。在根据最终X和Y矩阵结果,向用户进行推荐。

ALS的Spark实现

  Spark MLlib的ALS算法实现有点缺陷,要求用户和产品的ID必须是数值型,而且是32位非负整数。在计算以前应该首先检验一下数据量。

1)数据预处理

  过滤无效的用户艺术家ID和名字行,将格式不正确的数据行剔除掉。

def buildArtistByID(rawArtistData: Dataset[String]): DataFrame = {

  rawArtistData.flatMap { line =>

    val (id, name) = line.span(_ != '\t')

    if (name.isEmpty) {

      None

    } else {

      try {

        Some((id.toInt, name.trim))

      } catch {

        case _: NumberFormatException => None

      }

    }

  }.toDF("id", "name")

}

 

  过滤艺术家id和对应的别名id,将格式拼写错误的行剔除掉。

def buildArtistAlias(rawArtistAlias: Dataset[String]): Map[Int,Int] = {

  rawArtistAlias.flatMap { line =>

    val Array(artist, alias) = line.split('\t')

    if (artist.isEmpty) {

      None

    } else {

      Some((artist.toInt, alias.toInt))

    }

  }.collect().toMap

}

 

  将数据转换成Rating对象,Rating对象是ALS算法对“用户-产品-值”的抽象。

def buildCounts(

    rawUserArtistData: Dataset[String],

    bArtistAlias: Broadcast[Map[Int,Int]]): DataFrame = {

  rawUserArtistData.map { line =>

    val Array(userID, artistID, count) = line.split(' ').map(_.toInt)

    val finalArtistID = bArtistAlias.value.getOrElse(artistID, artistID)

    (userID, finalArtistID, count)

  }.toDF("user", "artist", "count")

}

 

2)模型构建

def model(

    rawUserArtistData: Dataset[String],

    rawArtistData: Dataset[String],

    rawArtistAlias: Dataset[String]): Unit = {

  val bArtistAlias = spark.sparkContext.broadcast(buildArtistAlias(rawArtistAlias))  //艺术家别名数据

  val trainData = buildCounts(rawUserArtistData, bArtistAlias).cache() //将数据转换成须要的格式

  val model = new ALS().

    setSeed(Random.nextLong()).

    setImplicitPrefs(true).

    setRank(10).

    setRegParam(0.01).

    setAlpha(1.0).

    setMaxIter(5).

    setUserCol("user").

    setItemCol("artist").

    setRatingCol("count").

    setPredictionCol("prediction").

    fit(trainData)

  trainData.unpersist()

  model.userFactors.select("features").show(truncate = false)

  val userID = 2093760

  val existingArtistIDs = trainData.

    filter($"user" === userID).

    select("artist").as[Int].collect()

  val artistByID = buildArtistByID(rawArtistData)

  artistByID.filter($"id" isin (existingArtistIDs:_*)).show()

  val topRecommendations = makeRecommendations(model, userID, 5)

  topRecommendations.show()

  val recommendedArtistIDs = topRecommendations.select("artist").as[Int].collect()

  artistByID.filter($"id" isin (recommendedArtistIDs:_*)).show()

  model.userFactors.unpersist()

  model.itemFactors.unpersist()

}

  本篇文章主要对ALS音乐推荐进行简单的介绍,下一篇会对模型的参数,以及模型的推荐效果进行评估,而且会对推荐结果进行优化。

 备注:若是文中排版出现错乱,请点击https://mp.weixin.qq.com/s/aqF38rDQdT35YrLAyLm-nA

更多精彩内容,欢迎扫码关注如下微信公众号:大数据技术宅。大数据、AI从关注开始
相关文章
相关标签/搜索