-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathword2vec.py
More file actions
353 lines (318 loc) · 18.3 KB
/
word2vec.py
File metadata and controls
353 lines (318 loc) · 18.3 KB
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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
# -*- coding:utf-8 -*-
import tensorflow as tf
import numpy as np
import math
import collections
import pickle as pkl
import jieba
import os
class word2vec():
# 1、初始化参数
def __init__(self,
vocab_list=None, # 词典默认为None,需用户传入
embedding_size=200, # 词向量维度,默认为200
win_len=3, # 单边窗口长,也就是上下文,考虑某个词的前面3个词,后面3个词
num_sampled=1000, # 负采样个数,默认1000
learning_rate=0.1, # 学习率
logdir='tmp/word2vec', # 模型保存目录
model_path="model" # 模型路径,用于训练好后加载模型用
):
# 获得模型的基本参数
if not os.path.exists(model_path):
os.makedirs(model_path)
self.batch_size = None # 一批中数据个数
if os.listdir(model_path): # 如果模型已经训练好,可以直接加载模型
self.load_model(model_path)
else:
# model parameters
assert type(vocab_list) == list
# 如果vocab_list类型不是list,即:type(vocab_list) == list为假,我们直接让程序出发异常
# 否则就可以继续运行
self.vocab_list = vocab_list # 词典
self.vocab_size = vocab_list.__len__() # 词典中词的总数
self.embedding_size = embedding_size # 词向量维度
self.win_len = win_len # 窗口大小
self.num_sampled = num_sampled # 负采样个数
self.learning_rate = learning_rate # 学习率
self.logdir = logdir # 目录
self.save_model_path= model_path
self.word2id = {} # word => id 的映射
for i in range(self.vocab_size):
self.word2id[self.vocab_list[i]] = i
self.train_words_num = 0 # 训练的单词对数
self.train_sents_num = 0 # 训练的句子数
self.train_times_num = 0 # 训练的次数(一次可以有多个句子)
self.train_loss_records = collections.deque(maxlen=10) # 保存最近10次的误差
self.train_loss_k10 = 0
self.build_graph() # 构建图
self.init_op() # 操作初始化
if os.listdir(model_path): # 模型路径不为None
tf_model_path = os.path.join(model_path, 'tf_vars') # 路径拼接
self.saver.restore(self.sess, tf_model_path) # 恢复以前保存的变量
# 2、图的定义
def build_graph(self):
self.graph = tf.Graph() # 声明图,所有的变量和操作都定义在图上,使用新构建的图
with self.graph.as_default(): # 定义graph图,在“graph”中定义运算和张量。
# 此时在这个代码块中,使用的就是新的定义的图graph(相当于把默认图换成了graph)
self.train_inputs = tf.placeholder(tf.int32, shape=[self.batch_size]) # 定义占位符,输入数据
self.train_labels = tf.placeholder(tf.int32, shape=[self.batch_size, 1]) # 中心词
self.embed_vector = tf.placeholder(tf.int32, shape=[self.vocab_size, 1]) # 中心词
# 定义词向量变量,形状为[vocab_size, embedding_size],并用正态分布均值为-1.0,方差为1.0的随机数赋值
# 其形状为[[],[],...,[]],行为每个词,列为每个词我们自定义的词向量维度
self.embedding_dict = tf.Variable(
tf.random_uniform([self.vocab_size, self.embedding_size], -1.0, 1.0)
)
# 定义权值变量,形状为[vocab_size, embedding_size],并用正态分布均值为0.0,方差为stddev的随机数赋值
self.nce_weight = tf.Variable(tf.truncated_normal([self.vocab_size, self.embedding_size],
stddev=1.0 / math.sqrt(self.embedding_size)))
# 定义偏置,形状为vocab_size行1列,初始化为0
self.nce_biases = tf.Variable(tf.zeros([self.vocab_size]))
# 将输入序列向量化
embed = tf.nn.embedding_lookup(self.embedding_dict, self.train_inputs) # batch_size
# 得到NCE损失
self.loss = tf.reduce_mean(
tf.nn.nce_loss(
weights=self.nce_weight,
biases=self.nce_biases,
labels=self.train_labels,
inputs=embed,
num_sampled=self.num_sampled,
num_classes=self.vocab_size
)
)
# tensorboard 相关
tf.summary.scalar('loss', self.loss) # 让tensorflow记录参数
# 根据 nce loss 来更新梯度和embedding,梯度下降法
self.train_op = tf.train.GradientDescentOptimizer(learning_rate=self.learning_rate).minimize(self.loss) # 训练操作
# 计算与指定若干单词的相似度
self.test_word_id = tf.placeholder(tf.int32, shape=[None])
vec_l2_model = tf.sqrt( # 求各词向量的L2模
tf.reduce_sum(tf.square(self.embedding_dict), 1, keep_dims=True)
# square:求平方
# reduce_sum:按行求和
# sqrt:求根号
)
avg_l2_model = tf.reduce_mean(vec_l2_model) # 求均值
tf.summary.scalar('avg_vec_model', avg_l2_model)
# generate embedd vector and similarity between words by words id
self.normed_embedding = self.embedding_dict / vec_l2_model # 对embedding_dict向量正则化
self.test_embed = tf.nn.embedding_lookup(self.normed_embedding, self.test_word_id) # 嵌入式张量列表中查找
self.similarity = tf.matmul(self.test_embed, self.normed_embedding, transpose_b=True)
# generate words id by embedd vector
tA_A = tf.matmul(tf.transpose(self.embedding_dict), self.embedding_dict)
tA_A_inv = tf.matrix_inverse(tA_A)
product = tf.matmul(tA_A_inv, tf.transpose(self.embedding_dict))
self.solution = tf.matmul(self.embed_vector, product)
# matmul:矩阵相乘
# 变量初始化
self.init = tf.global_variables_initializer()
# 合并默认图中收集的所有摘要
self.merged_summary_op = tf.summary.merge_all()
# 保存模型
self.saver = tf.train.Saver()
# 3、操作初始化
def init_op(self):
# 创建会话
self.sess = tf.Session(graph=self.graph)
# 执行会话
self.sess.run(self.init)
# FileWriter类提供了一个机制来创建指定目录的事件文件,并添加摘要和事件给它
self.summary_writer = tf.summary.FileWriter(self.logdir, self.sess.graph)
# 4、训练模型
def train_by_sentence(self, input_sentence=[] , epoch=0):
# input_sentence: [sub_sent1, sub_sent2, ...]
# 每个sub_sent是一个单词序列,例如['这次','大选','让']
# sent_num = input_sentence.__len__()
batch_inputs = [] # 输入数据,上下文
batch_labels = [] # 中心词
for sent in input_sentence: # 遍历每个sub_sent
for i in range(sent.__len__()): # 遍历sub_sent中每个词
# 窗口
start = max(0, i - self.win_len) # 开始下标 窗口为 [-win_len,+win_len],总计长2*win_len+1
end = min(sent.__len__(), i + self.win_len + 1) # 结束下标
for index in range(start, end): # 遍历窗口内的词
if index == i: # 跳过中心词
continue
else:
input_id = self.word2id.get(sent[i]) # 上下文
label_id = self.word2id.get(sent[index]) # 中心词
if not (input_id and label_id):
continue
batch_inputs.append(input_id) # 上下文id添加到batch_inputs
batch_labels.append(label_id) # 中心词id添加到batch_labels
if len(batch_inputs) == 0: # 直到batch_inputs为空,结束函数
return
batch_inputs = np.array(batch_inputs, dtype=np.int32) # 转为数组
batch_labels = np.array(batch_labels, dtype=np.int32)
batch_labels = np.reshape(batch_labels, [batch_labels.__len__(), 1]) # 改变形状
# 模型需传入的输入数据
feed_dict = {
self.train_inputs: batch_inputs,
self.train_labels: batch_labels
}
# 开启会话,并传入数据
_, loss_val, summary_str = self.sess.run([self.train_op, self.loss, self.merged_summary_op],
feed_dict=feed_dict)
# 训练损失
self.train_loss_records.append(loss_val)
# self.train_loss_k10 = sum(self.train_loss_records)/self.train_loss_records.__len__()
self.train_loss_k10 = np.mean(self.train_loss_records)
if self.train_sents_num % 1000 == 0:
# add_summary:FileWriter类中的函数,将摘要协议缓冲区添加到事件文件中
self.summary_writer.add_summary(summary_str, self.train_sents_num)
print("{a} sentences dealed, loss: {b} , Epoch:{num}"
.format(a=self.train_sents_num, b=self.train_loss_k10, num=epoch))
# 训练次数
self.train_words_num += batch_inputs.__len__()
self.train_sents_num += input_sentence.__len__()
self.train_times_num += 1
# 5、计算相似概率,测试词的id,只选取相似概率前10个
def cal_similarity(self, test_word_id_list, top_k=10):
# 相似度矩阵
sim_matrix, embed,solution = self.sess.run([self.similarity, self.test_embed,self.solution], feed_dict={self.test_word_id: test_word_id_list})
sim_mean = np.mean(sim_matrix) # 按行,求均值
sim_var = np.mean(np.square(sim_matrix - sim_mean)) # 按行求方差
test_words = []
near_words = []
for i in range(test_word_id_list.__len__()):
test_words.append(self.vocab_list[test_word_id_list[i]]) # 测试单词
nearst_id = (-sim_matrix[i, :]).argsort()[1:top_k + 1] # 最相似的10个词的id,并排序
nearst_word = [self.vocab_list[x] for x in nearst_id] # 最相似的词
near_words.append(nearst_word) # 添加到相似词near_words列表中
return test_words, near_words, sim_mean, sim_var, embed,solution # 返回测试词,相似词,相似度均值,相似度方差
# 6、模型保存
def save_model(self):
if os.path.isfile(self.save_model_path):
raise RuntimeError('the save path should be a dir')
if not os.path.exists(self.save_model_path):
os.mkdir(self.save_model_path)
# 记录模型各参数
model = {}
var_names = ['vocab_size', # int model parameters
'vocab_list', # list
'learning_rate', # int
'word2id', # dict
'embedding_size', # int
'logdir', # str
'win_len', # int
'num_sampled', # int
'train_words_num', # int train info
'train_sents_num', # int
'train_times_num', # int
'train_loss_records', # int train loss
'train_loss_k10', # int
]
for var in var_names:
model[var] = eval('self.' + var)
param_path = os.path.join(self.save_model_path, 'params.pkl') # 路径拼接
if os.path.exists(param_path): # 如果路径存在就删除
os.remove(param_path)
with open(param_path, 'wb') as f: # 否则以'wb'方式打开文件
pkl.dump(model, f) # 保存模型
# 记录tf模型
tf_path = os.path.join(self.save_model_path, 'tf_vars')
if os.path.exists(tf_path):
os.remove(tf_path)
self.saver.save(self.sess, tf_path)
# 7、模型加载
def load_model(self, model_path):
if not os.path.exists(model_path): # 如果不存在,即没有训练好的模型
raise RuntimeError('file not exists')
# 否则拼接路径
param_path = os.path.join(model_path, 'params.pkl')
with open(param_path, 'rb') as f: # 以'rb'方式打开文件,读取参数
model = pkl.load(f)
self.vocab_list = model['vocab_list']
self.vocab_size = model['vocab_size']
self.logdir = model['logdir']
self.word2id = model['word2id']
self.embedding_size = model['embedding_size']
self.learning_rate = model['learning_rate']
self.win_len = model['win_len']
self.num_sampled = model['num_sampled']
self.train_words_num = model['train_words_num']
self.train_sents_num = model['train_sents_num']
self.train_times_num = model['train_times_num']
self.train_loss_records = model['train_loss_records']
self.train_loss_k10 = model['train_loss_k10']
if __name__ == '__main__':
if os.path.exists("tmp/test.txt") and os.listdir("model"):
# 创建模型,训练
with open("tmp/test.txt", "rb") as fp: # Pickling
unpickler = pkl.Unpickler(fp)
try:
word_list = unpickler.load()
except EOFError:
word_list = list()
w2v = word2vec(vocab_list=word_list, # 词典集
embedding_size=200, # 词向量的维度
win_len=2, # 窗口大小
learning_rate=0.1, # 学习率
num_sampled=100, # 负采样个数
)
w2v.load_model("model") # 加载模型,得到保存的模型参数
else:
# step 1 读取停用词,构建停用词set集合
stop_words = []
with open('stop_words.txt', encoding='utf-8') as f: # 读取stop_words.txt,每行为一个词
line = f.readline() # 按行读取
while line:
stop_words.append(line[:-1]) # 去除每行后面的换行符并添加到stop_words中
line = f.readline()
stop_words = set(stop_words) # 通过set函数去除重复词
print('停用词读取完毕,共{n}个单词'.format(n=len(stop_words)))
# step2 读取文本,预处理,分词,得到词典
all_word_list = [] # 用于记录所有的词,即词典
sentence_list = [] # 用于记录所有样本的样本list,如:[["今天","天气","不错"],["我","是","谁"],...,["","",""]]
line_sum = 1
with open('280.txt', encoding='gbk') as f:
line = f.readline() # 按行读取
while line: # 判断有没有读取到文件末尾,没有就执行循环语句
print("正在处理第", line_sum, "行")
while '\n' in line:
line = line.replace('\n', '') # 将'\n'替换为''
while ' ' in line:
line = line.replace(' ', '') # 将' '替换为''
if len(line) > 0: # 如果句子非空
raw_words = list(jieba.cut(line, cut_all=False)) # 使用jieba分词
dealed_words = [] # 用于记录去除停用词以及规定词后的词
for word in raw_words:
# 如果词不属于停用词且不是['qingkan520', 'www', 'com', 'http']中的词
if word not in stop_words and word not in ['qingkan520', 'www', 'com', 'http']:
all_word_list.append(word) # 将该词添加到样本的词list中
dealed_words.append(word) # 将该词添加到处理后的词list中
sentence_list.append(dealed_words) # 每行为一个样本,并将处理过的词list添加到样本list
line = f.readline() # 读取下一行
line_sum += 1
word_count = collections.Counter(all_word_list) # 计数器,返回dict类型,{"单词1":15,"单词2":5,"单词3":2}
# 表示单词1出现了15次,单词2出现了5次,单词3出现了2次,依次类推
print('文本中总共有{n1}个单词,不重复单词数{n2},选取前30000个单词进入词典'
.format(n1=len(all_word_list), n2=len(word_count)))
word_count = word_count.most_common(30000) # 返回一个TopN列表,如:[("单词1",15),("单词2",5),("单词3",2)]
word_list = [x[0] for x in word_count] # 取出TopN列表中的单词,放到word_list中,构成["单词1","单词2","单词3"]
with open("tmp/test.txt", "wb") as fp: # Pickling
pkl.dump(word_list, fp)
# 创建模型,训练
w2v = word2vec(vocab_list=word_list, # 词典集
embedding_size=200, # 词向量的维度
win_len=2, # 窗口大小
learning_rate=0.1, # 学习率
num_sampled=100, # 负采样个数
)
num_steps = 3 # 训练次数
for i in range(num_steps):
for j in range(len(sentence_list)):
# print (i%len(sentence_list))
sent = sentence_list[j]
# 如果len(sentence_list) < num_steps,会有重复训练的行,如num_steps=20,len(sentence_list)=10
# 0 % 10 = 0,1 % 10 = 1,...,10 % 10 = 0,11 % 10 = 1,12 % 10 = 2...
# 这样有些就被训练多次了
w2v.train_by_sentence([sent], epoch=i) # 按行训练模型,我们认为一行代表一个样本
w2v.save_model() # 保存模型,加训练好的模型参数保存
# 测试
test_word = ["萧炎"]
test_id = [word_list.index(x) for x in test_word] # 测试单词在词典word_list中的索引,即为单词id
test_words, near_words, sim_mean, sim_var, embed, wordid = w2v.cal_similarity(test_id) # 计算相似度
# print("Test_words:{}, near_words: {}, sim_mean:{}, sim_var: {}, embed: {}"
# .format(test_words, near_words, sim_mean, sim_var,embed))
print(wordid)