-
Notifications
You must be signed in to change notification settings - Fork 19
/
pmf.py
73 lines (61 loc) · 2.97 KB
/
pmf.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
import numpy as np
import random
from data import *
from evaluation import *
class pmf():
def __init__(self,
train_list, # train_list: train data
test_list, # test_list: test data
N, # N:the number of user
M, # M:the number of item
K=10, # K: the number of latent factor
learning_rate=0.001, # learning_rate: the learning rata
lamda_regularizer=0.1, # lamda_regularizer: regularization parameters
max_iteration=50 # max_iteration: the max iteration
):
self.train_list = train_list
self.test_list = test_list
self.N = N
self.M = M
self.K = K
self.learning_rate = learning_rate
self.lamda_regularizer = lamda_regularizer
self.max_iteration = max_iteration
def train(self):
P = np.random.normal(0, 0.1, (self.N, self.K))
Q = np.random.normal(0, 0.1, (self.M, self.K))
train_mat = sequence2mat(sequence = self.train_list, N = self.N, M = self.M)
test_mat = sequence2mat(sequence = self.test_list, N = self.N, M = self.M)
records_list = []
for step in range(self.max_iteration):
los=0.0
for data in self.train_list:
u,i,r = data
P[u],Q[i],ls = self.update(P[u], Q[i], r=r,
learning_rate=self.learning_rate,
lamda_regularizer=self.lamda_regularizer)
los += ls
pred_mat = self.prediction(P,Q)
mae, rmse, recall, precision = evaluation(pred_mat, train_mat, test_mat)
records_list.append(np.array([los, mae, rmse, recall, precision]))
if step % 10 ==0:
print(' step:%d \n loss:%.4f,mae:%.4f,rmse:%.4f,recall:%.4f,precision:%.4f'
%(step,los,mae,rmse,recall,precision))
print(' end. \n loss:%.4f,mae:%.4f,rmse:%.4f,recall:%.4f,precision:%.4f'
%(records_list[-1][0],records_list[-1][1],records_list[-1][2],records_list[-1][3],records_list[-1][4]))
return P, Q, np.array(records_list)
def update(self, p, q, r, learning_rate=0.001, lamda_regularizer=0.1):
error = r - np.dot(p, q.T)
p = p + learning_rate*(error*q - lamda_regularizer*p)
q = q + learning_rate*(error*p - lamda_regularizer*q)
loss = 0.5 * (error**2 + lamda_regularizer*(np.square(p).sum() + np.square(q).sum()))
return p, q, loss
def prediction(self, P, Q):
N,K = P.shape
M,K = Q.shape
rating_list=[]
for u in range(N):
u_rating = np.sum(P[u,:]*Q, axis=1)
rating_list.append(u_rating)
r_pred = np.array(rating_list)
return r_pred