使用奇异值分解求解普通最小二乘问题的玩具R函数

3

我正在尝试编写使用奇异值分解矩阵的多元回归分析函数(y = Xb + e)。输入为yX,回归系数向量b,残差向量e和可解释方差R2作为输出。以下是我到目前为止所写的内容,但我完全被卡住了。权重的labels部分也给出错误信息。这个labels部分是什么?有人可以给我一些提示来帮助我继续吗?

Test <- function(X, y) {
  x <- t(A) %*% A
  duv <- svd(x)
  x.inv <- duv$v %*% diag(1/duv$d) %*% t(duv$u)
  x.pseudo.inv <- x.inv %*% t(A)
  w <- x.pseudo.inv %*% labels
  return(b, e, R2)
  }

大小写敏感 - xX 是不同的。你函数的第一行使用了 A,但是 A 不是输入的。它从哪里来的?我也不确定 labels 是什么 - 这不是你写的代码吗?它应该做什么? - Gregor Thomas
这是我在这个网站上找到的大部分代码。他们使用了标签。是的,抱歉,A是一个测试矩阵。不过我的问题已经在下面得到了答复,还是谢谢! - Rilja
1个回答

7
你离题了...奇异值分解应用于模型矩阵X而非正常矩阵X'X。以下是正确的步骤:

svd for ols

因此,在编写R函数时,我们应该这样做:

svdOLS <- function (X, y) {
  SVD <- svd(X)
  V <- SVD$v
  U <- SVD$u
  D <- SVD$d
  ## regression coefficients `b`
  ## use `crossprod` for `U'y`
  ## use recycling rule for row rescaling of `U'y` by `D` inverse
  ## use `as.numeric` to return vector instead of matrix
  b <- as.numeric(V %*% (crossprod(U, y) / D))
  ## residuals
  r <- as.numeric(y - X %*% b)
  ## R-squared
  RSS <- crossprod(r)[1]
  TSS <- crossprod(y - mean(y))[1]
  R2 <- 1 - RSS / TSS
  ## multiple return via a list
  list(coefficients = b, residuals = r, R2 = R2)
  }

让我们进行一次测试

## toy data
set.seed(0)
x1 <- rnorm(50); x2 <- rnorm(50); x3 <- rnorm(50); y <- rnorm(50)
X <- model.matrix(~ x1 + x2 + x3)

## fitting linear regression: y ~ x1 + x2 + x3
svdfit <- svdOLS(X, y)

#$coefficients
#[1]  0.14203754 -0.05699557 -0.01256007  0.09776255
#
#$residuals
# [1]  1.327108410 -1.400843739 -0.071885339  2.285661880  0.882041795
# [6] -0.535230752 -0.927750996  0.262674650 -0.133878558 -0.559783412
#[11]  0.264204296 -0.581468657  2.436913000  1.517601798  0.774515419
#[16]  0.447774149 -0.578988327  0.664690723 -0.511052627 -1.233302697
#[21]  1.740216739 -1.065592673 -0.332307898 -0.634125164 -0.975142054
#[26]  0.344995480 -1.748393187 -0.414763742 -0.680473508 -1.547232557
#[31] -0.383685601 -0.541602452 -0.827267878  0.894525453  0.359062906
#[36] -0.078656943  0.203938750 -0.813745178 -0.171993018  1.041370294
#[41] -0.114742717  0.034045040  1.888673004 -0.797999080  0.859074345
#[46]  1.664278354 -1.189408794  0.003618466 -0.527764821 -0.517902581
#
#$R2
#[1] 0.008276773

另一方面,我们可以使用 .lm.fit 来检查正确性:

qrfit <- .lm.fit(X, y)

这与系数和残差完全相同:

all.equal(svdfit$coefficients, qrfit$coefficients)
# [1] TRUE

all.equal(svdfit$residuals, qrfit$residuals)
# [1] TRUE

好的,我明白你的意思了,谢谢!但我该如何从中获取 b、e 和 R^2 呢? - Rilja
非常感谢!你救了我的一天/晚上。如果可以的话,你能否帮忙看看我和同学遇到的类似问题吗?https://dev59.com/TJvga4cB1Zd3GeqP9fLi - Rilja

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接