主成分分析

1 主成分分析原理

  主成分分析是最常用的一种降维方法。我们首先考虑一个问题:对于正交矩阵空间中的样本点,如何用一个超平面对所有样本进行恰当的表达。容易想到,如果这样的超平面存在,那么他大概应该具有下面的性质。

  • 最近重构性:样本点到超平面的距离都足够近

  • 最大可分性:样本点在这个超平面上的投影尽可能分开

  基于最近重构性和最大可分性,能分别得到主成分分析的两种等价推导。

1.1 最近重构性

  假设我们对样本点进行了中心化,即所有样本的和为0。再假设投影变换后得到的新坐标系为:

1.1

  若丢弃新坐标系中的部分坐标,将维度降到d',则样本点$x{i}$在低位坐标系中的投影是$z{i}$ :

1.2

  这里$z{ij}$是$x{i}$在低维坐标系下第j维的坐标。若基于$z{i}$来重构$x{i}$ ,那么可以得到

1.3

  考虑整个训练集,原样本点和基于投影重构的样本点之间的距离为

1.4

  根据最近重构性,最小化上面的式子,就可以得到主成分分析的优化目标

1.5

1.2 最大可分性

  从最大可分性出发,我们可以得到主成分分析的另一种解释。我们知道,样本点$x{i}$在新空间中超平面上的投影是$W^{T}x{i}$ ,
若所有样本点的投影能尽可能分开,则应该使投影后样本点的方差最大化。投影后样本点的方差是

1.6

  于是优化目标可以写为

1.7

  这个优化目标和上文的优化目标是等价的。对优化目标使用拉格朗日乘子法可得

1.8

  于是,只需要对协方差矩阵进行特征值分解,将得到的特征值排序,在取前d'个特征值对应的特征向量,即得到主成分分析的解。

2 源码分析

2.1 实例

  1. import org.apache.spark.mllib.linalg.Matrix
  2. import org.apache.spark.mllib.linalg.distributed.RowMatrix
  3. val mat: RowMatrix = ...
  4. // Compute the top 10 principal components.
  5. val pc: Matrix = mat.computePrincipalComponents(10) // Principal components are stored in a local dense matrix.
  6. // Project the rows to the linear space spanned by the top 10 principal components.
  7. val projected: RowMatrix = mat.multiply(pc)

2.2 实现代码

  主成分分析的实现代码在RowMatrix中实现。源码如下:

  1. def computePrincipalComponents(k: Int): Matrix = {
  2. val n = numCols().toInt
  3. //计算协方差矩阵
  4. val Cov = computeCovariance().toBreeze.asInstanceOf[BDM[Double]]
  5. //特征值分解
  6. val brzSvd.SVD(u: BDM[Double], _, _) = brzSvd(Cov)
  7. if (k == n) {
  8. Matrices.dense(n, k, u.data)
  9. } else {
  10. Matrices.dense(n, k, Arrays.copyOfRange(u.data, 0, n * k))
  11. }
  12. }

  这段代码首先会计算样本的协方差矩阵,然后在通过breezesvd方法进行奇异值分解。这里由于协方差矩阵是方阵,所以奇异值分解等价于特征值分解。下面是计算协方差的代码

  1. def computeCovariance(): Matrix = {
  2. val n = numCols().toInt
  3. checkNumColumns(n)
  4. val (m, mean) = rows.treeAggregate[(Long, BDV[Double])]((0L, BDV.zeros[Double](n)))(
  5. seqOp = (s: (Long, BDV[Double]), v: Vector) => (s._1 + 1L, s._2 += v.toBreeze),
  6. combOp = (s1: (Long, BDV[Double]), s2: (Long, BDV[Double])) =>
  7. (s1._1 + s2._1, s1._2 += s2._2)
  8. )
  9. updateNumRows(m)
  10. mean :/= m.toDouble
  11. // We use the formula Cov(X, Y) = E[X * Y] - E[X] E[Y], which is not accurate if E[X * Y] is
  12. // large but Cov(X, Y) is small, but it is good for sparse computation.
  13. // TODO: find a fast and stable way for sparse data.
  14. val G = computeGramianMatrix().toBreeze.asInstanceOf[BDM[Double]]
  15. var i = 0
  16. var j = 0
  17. val m1 = m - 1.0
  18. var alpha = 0.0
  19. while (i < n) {
  20. alpha = m / m1 * mean(i)
  21. j = i
  22. while (j < n) {
  23. val Gij = G(i, j) / m1 - alpha * mean(j)
  24. G(i, j) = Gij
  25. G(j, i) = Gij
  26. j += 1
  27. }
  28. i += 1
  29. }
  30. Matrices.fromBreeze(G)
  31. }

参考文献

【1】 机器学习.周志华