forked from dlaptev/RobustPCA
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRobustPCA.m
60 lines (53 loc) · 1.71 KB
/
RobustPCA.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
function [L, S] = RobustPCA(X, lambda, mu, tol, max_iter)
% - X is a data matrix (of the size N x M) to be decomposed
% X can also contain NaN's for unobserved values
% - lambda - regularization parameter, default = 1/sqrt(max(N,M))
% - mu - the augmented lagrangian parameter, default = 10*lambda
% - tol - reconstruction error tolerance, default = 1e-6
% - max_iter - maximum number of iterations, default = 1000
[M, N] = size(X);
unobserved = isnan(X);
X(unobserved) = 0;
normX = norm(X, 'fro');
% default arguments
if nargin < 2
lambda = 1 / sqrt(max(M,N));
end
if nargin < 3
mu = 10*lambda;
end
if nargin < 4
tol = 1e-6;
end
if nargin < 5
max_iter = 1000;
end
% initial solution
L = zeros(M, N);
S = zeros(M, N);
Y = zeros(M, N);
for iter = (1:max_iter)
% ADMM step: update L and S
L = Do(1/mu, X - S + (1/mu)*Y);
S = So(lambda/mu, X - L + (1/mu)*Y);
% and augmented lagrangian multiplier
Z = X - L - S;
Z(unobserved) = 0; % skip missing values
Y = Y + mu*Z;
err = norm(Z, 'fro') / normX;
if (iter == 1) || (mod(iter, 10) == 0) || (err < tol)
fprintf(1, 'iter: %04d\terr: %f\trank(L): %d\tcard(S): %d\n', ...
iter, err, rank(L), nnz(S(~unobserved)));
end
if (err < tol) break; end
end
end
function r = So(tau, X)
% shrinkage operator
r = sign(X) .* max(abs(X) - tau, 0);
end
function r = Do(tau, X)
% shrinkage operator for singular values
[U, S, V] = svd(X, 'econ');
r = U*So(tau, S)*V';
end