-
Notifications
You must be signed in to change notification settings - Fork 1
/
decoder.py
116 lines (92 loc) · 3.66 KB
/
decoder.py
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
108
109
110
111
112
113
114
115
116
import numpy as np
from numpy.linalg import svd
from utils import disc_pr
from scipy.stats import pearsonr
def rank_classification(target_vector,candidate_vectors,id):
'''
:param target_vector: 1*n
:param candidate_vectors: m*n
:param id :integer correct prediction
:return: rank classification for target in [0,1]
'''
corr = []
cds = candidate_vectors.shape[0]
for i in range(cds):
corr.append((pearsonr(candidate_vectors[i,:],target_vector),i))
corr.sort(key=lambda tup: tup[0])
pos = 0
for i,j in enumerate(corr):
if j[tuple[1]] == id:
pos=i
break
assert pos
rank_clas = abs(pos - cds)/cds
return rank_clas
def regression_decoder(train_data,train_targets):
'''
:param train_data: #examples x #voxels matrix
:param train_targets: # examples x #dimensions matrix
:return:
weighMatrix - a #voxels+1 x #dimensions weight matrix
r - #dimensions vector with the regularization parameter
value for each dimension
column i of weightMatrix has #voxels weights + intercept (last row)
for predicting target i
This function uses an efficient implementation of cross-validation within the
training set to pick a different optimal value of the
regularization parameter for each semantic dimension in the target vector
This function uses kernel ridge regression with a linear
kernel. This allows us to use the full brain as input features
because we avoid the large inversion of the voxels/voxels matrix
'''
# add one in the end of train data for the bias term
h_x = np.ones((train_data.shape[0], train_data.shape[1] + 1))
h_x[:, :-1] = train_data
train_data = h_x
dims_vxl = train_data.shape[1]
emb_dim = train_targets.shape[1]
examples = train_data.shape[0]
assert train_data.shape[0]==train_targets.shape[0],'Same numbers of examples for data and targets '
params = [1, .5, 5, 0.1, 10, 0.01, 100, 0.001, 1000, 0.0001, 10000, 0.00001,
100000, 0.000001, 1000000]
n_words = train_data.shape[0]
cv_err = np.zeros((len(params), emb_dim))
K = np.matmul(train_data, train_data.T)
U,D,V = svd(K)
D = np.eye(U.shape[1], V.shape[0])*D
for idx,reg_param in enumerate(params):
dlambda = D + reg_param*np.eye(D.shape[0], D.shape[1])
dlambdaInv = np.diag(1/np.diag(dlambda))
klambdainv = np.matmul(np.matmul(V,dlambdaInv), U.T)
K_p = np.matmul(train_data.T, klambdainv)
S = np.matmul(train_data, K_p)
weights = np.matmul(K_p, train_targets)
# Snorm = repmat(1 - diag(S), 1, train_targets.shape[1])
Snorm = np.tile((1-np.diag(S)).reshape(np.diag(S).shape[0],1),(1,emb_dim))
#Snorm = np.tile(1-np.diag(S),(1,emb_dim))
#print(Snorm.shape)
Y_diff = train_targets - np.matmul(train_data, weights)
disc_pr()
disc_pr()
print(Y_diff)
disc_pr()
disc_pr()
Y_diff = Y_diff/Snorm
print(Y_diff)
disc_pr()
disc_pr()
cv_err[idx, :]=(1/examples)*np.sum(Y_diff*Y_diff) # elementwise
disc_pr()
print (cv_err)
disc_pr()
minerridx = cv_err.argmin(axis=0)
# minerr = np.amin(cv_err)
reg_dim = np.zeros((1, emb_dim))
for i in range(emb_dim):
reg_param = params[minerridx[i]]
reg_dim[0,i]=reg_param
dlambda = D + reg_param*np.eye(D.shape[0], D.shape[1])
dlambdaInv = np.diag(1/np.diag(dlambda))
klambdainv = np.matmul(np.matmul(V, dlambdaInv), U.T)
weights[:,i] = np.matmul(np.matmul(train_data.T, klambdainv), train_targets[:,i])
return weights,reg_dim