-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathBTL.py
39 lines (36 loc) · 1.3 KB
/
BTL.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
'''
computes the parameters using maximum likelihood principle.
This function is adapted from the Matlab version provided by David Hunter
http://personal.psu.edu/drh20/code/btmatlab
'''
# input:
# Data: nxn comparison matrix
# probs: when True, turns the comparison matrix into pairwise preference probabilities, i.e., Data[i,j] + Data[j,i] = 1
# max_iter: the maximum number of iterations
# output: the estimated parameters
def BTL(Data, probs=False, max_iter=10**5):
wm = Data
if probs:
np.fill_diagonal(wm, 1)
wm = wm/(wm+wm.T)
np.fill_diagonal(wm, 0)
n = wm.shape[0]
nmo = n-1
pi = np.ones(nmo, dtype=float)
gm = (wm[:,range(nmo)]).T + wm[range(nmo),:]
wins = np.sum(wm[range(nmo),], axis=1)
gind = gm>0
z = np.zeros((nmo,n))
pisum = z
for _ in range(max_iter):
pius = np.repeat(pi, n).reshape(nmo, -1)
piust = (pius[:,range(nmo)]).T
piust = np.column_stack((piust, np.repeat(1,nmo)))
pisum[gind] = pius[gind]+piust[gind]
z[gind] = gm[gind] / pisum[gind]
newpi = wins / np.sum(z, axis=1)
if np.linalg.norm(newpi - pi, ord=np.inf) <= 1e-6:
newpi = np.append(newpi, 1)
return newpi/sum(newpi)
pi = newpi
raise RuntimeError('did not converge')