forked from jduffield65/iss
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathomp_free_background.m
107 lines (100 loc) · 3.34 KB
/
omp_free_background.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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
function [x,r,r_norm] = omp_free_background(A,b,n_nonzero_coefs,...
thresh,background_indices,gene_indices)
%% [x,r,r_norm] = omp_free_background(A,b,n_nonzero_coefs,...
% thresh,background_indices,gene_indices)
% Extended from https://github.com/seunghwanyoo/omp
% Input:
% A: dictionary (matrix) such that norm(A(:,i))=1 for all i.
% b: signal
% n_nonzero_coefs: max number of atoms from A (not including background)
% that can be selected.
% thresh: OMP stops when reduction in residue drops below thresh.
% background_indices: atoms at these indices will be selected first.
% gene_indices: all atoms in dictionary not background.
% Output:
% x: coeff vector for sparse representation
% r: residual
% r_norm: norm of residual
%
% This explains the signal, b, with all atoms indicated by
% background_indices.
% It then fits atoms indicated by gene_indices until the residual reduction
% falls below thresh or more than n_nonzero_coefs atoms have been added.
[N,K] = size(A); % N:dim of signal, K:#atoms in dictionary
if (N ~= size(b))
error('Dimension not matched');
end
if nargin<3 || isempty(n_nonzero_coefs)
n_nonzero_coefs=round(K/10);
end
if nargin<4 || isempty(thresh)
thresh=0;
end
if nargin<5 || isempty(background_indices)
background_indices=[];
end
if nargin<6 || isempty(gene_indices)
gene_indices = quick_setdiff(1:K,background_indices);
end
S_background = length(background_indices);
S = S_background+n_nonzero_coefs; % sparsity level
x = zeros(K,1); % coefficient (output)
r = b; % residual of b
r_norm = norm(b); % norm of residual of b
r_norm_last = inf; % inf so have to have atleast one coefficient.
omega = zeros(S,1); % selected support
omega_last = omega;
%A_omega = []; % corresponding columns of A
cnt = 0;
while (cnt < S) % choose S atoms
cnt = cnt+1;
if cnt<=S_background
ichosen = background_indices(cnt);
else
%x_tmp = zeros(K,1);
if cnt>1
gene_indices = gene_indices(gene_indices~=omega(cnt-1));
end
%inds = quick_setdiff(1:K,omega(omega~=0)); % iterate all columns except for the chosen ones
% for i = gene_indices
% %x_tmp(i) = A(:,i)' * r / norm(A(:,i)); % sol of min ||a'x-b||
% x_tmp(i) = A(:,i)' * r; % sol of min ||a'x-b||
% end
x_tmp = A'*r; % sol of min ||a'x-b||
[~,ichosen] = max(abs(x_tmp(gene_indices))); % choose the maximum
ichosen = gene_indices(ichosen);
end
omega(cnt) = ichosen;
%A_omega = [A_omega A(:,ichosen)];
A_omega = A(:,omega(1:cnt));
x_ls = A_omega \ b; % Aomega * x_ls = b
r = b - A_omega * x_ls; % update r
r_norm = norm(r);
if r_norm_last-r_norm<thresh && cnt>S_background
%If met residue threshold, go back to last solution
%i.e. current solution is too many atoms.
x_ls = x_ls_last;
omega = omega_last(1:cnt-1);
r_norm = r_norm_last;
S = cnt-1;
break;
else
x_ls_last = x_ls;
omega_last = omega;
r_norm_last = r_norm;
end
end
for i = 1:S
x(omega(i)) = x_ls(i); %x_sparse(i).value;
end
end
function Z = quick_setdiff(X,Y)
if ~isempty(X)&&~isempty(Y)
check = false(1, max(max(X), max(Y)));
check(X) = true;
check(Y) = false;
Z = X(check(X));
else
Z = X;
end
end