TensorFlow訓(xùn)練詞向量

參考: https://www.tensorflow.org/tutorials/word2vec

官網(wǎng)的這個(gè)教程主要講word2vec的skip-gram模型,沒(méi)有講CBOW,并且訓(xùn)練用的負(fù)采樣,沒(méi)有用層次softmax。

動(dòng)機(jī)

動(dòng)機(jī)其實(shí)就是分布式假說(shuō):

在相同上下文中的詞有相似語(yǔ)義(words that appear in the same contexts share semantic meaning)

根據(jù)分布式假說(shuō)表示詞的方法分為兩類:

  • count-based methods (e.g. Latent Semantic Analysis)
  • predictive methods (e.g. neural probabilistic language models)

預(yù)測(cè)方法的代表就是Word2Vec模型,有兩種:

  • Continuous Bag-of-Words model (CBOW)
  • Skip-Gram model

CBOW 從上下文預(yù)測(cè)目標(biāo)詞,skip-gram正好相反,從目標(biāo)詞預(yù)測(cè)上下文中的詞。

CBOW在許多分布式信息上進(jìn)行平滑(將整個(gè)上下文作為一種情況),大多數(shù)情況下這個(gè)模型在小一點(diǎn)的數(shù)據(jù)集上更有效??墒?,skip-gram將每一對(duì)context-target詞作為一種新的情況,在大一點(diǎn)的數(shù)據(jù)集上的會(huì)有更好效果。

噪聲對(duì)比訓(xùn)練

神經(jīng)概率語(yǔ)言模型通常用最大似然估計(jì)來(lái)訓(xùn)練,即最大化給定輸入h(歷史信息,前n-1個(gè)詞),輸出下一個(gè)詞wt的概率,使用softmax函數(shù),取log后就是我們想要的準(zhǔn)則函數(shù),叫作log-likelihood。這個(gè)值的計(jì)算代價(jià)非常高,因?yàn)?strong>softmax的分母要計(jì)算整個(gè)詞表的得分。

訓(xùn)練數(shù)據(jù)的構(gòu)造方法則為:對(duì)于每一個(gè)ngram片段,前n-1個(gè)詞為構(gòu)成輸入數(shù)據(jù)(詞向量拼接),第n個(gè)詞構(gòu)成輸出類標(biāo)。由于輸出是一個(gè)softmax層,則對(duì)應(yīng)詞表大小個(gè)輸出,如果模型訓(xùn)練ok的話,那么這里的第n個(gè)詞對(duì)應(yīng)的位置輸出概率應(yīng)該最大。

在word2vec中,不再需要計(jì)算整個(gè)詞表每個(gè)詞的得分。CBOW和skip-gram模型的核心是訓(xùn)練一個(gè)二元分類器,將目標(biāo)詞從k個(gè)構(gòu)造的noise words區(qū)分出來(lái)。CBOW的模型圖如下,skip-gram類似只是反過(guò)來(lái)。

此時(shí)目標(biāo)函數(shù)就變?yōu)樵诋?dāng)前上下文h下,目標(biāo)詞為1的概率log值,加上k個(gè)噪聲詞為0的概率log值的期望。在實(shí)踐中,我們從噪聲分布中采樣k個(gè)詞來(lái)近似計(jì)算期望。

這個(gè)目標(biāo)可以看做是計(jì)算一個(gè)最優(yōu)模型,這個(gè)模型賦予真實(shí)詞高概率,賦予噪聲詞低概率。學(xué)術(shù)上,這個(gè)叫做負(fù)采樣。這個(gè)方法使得訓(xùn)練變得非常有效,因?yàn)楝F(xiàn)在計(jì)算損失函數(shù)只需要考慮k個(gè)噪聲詞,而不是整個(gè)詞表。在TensorFlow中,有一個(gè)非常相似的損失函數(shù)tf.nn.nce_loss()。

Skip-gram模型

舉個(gè)例子說(shuō)明訓(xùn)練的過(guò)程。

例子為:

the quick brown fox jumped over the lazy dog

上下文可以是語(yǔ)法詞法等,這里定義為左邊的詞和右邊的詞,窗口大小設(shè)置為1,則可以得到context-target訓(xùn)練對(duì)如下:

([the, brown], quick), ([quick, fox], brown), ([brown, jumped], fox), ...

即通過(guò)quick預(yù)測(cè)the和brown, 從brown預(yù)測(cè)quick和fox,這樣數(shù)據(jù)集變?yōu)?

(quick, the), (quick, brown), (brown, quick), (brown, fox), ...

目標(biāo)函數(shù)是定義在整個(gè)數(shù)據(jù)集上的,但是實(shí)際訓(xùn)練以minibatch為單位計(jì)算,batch_size一般為16 <= batch_size <= 512。

想象一下訓(xùn)練過(guò)程,假設(shè)當(dāng)前觀察到上面第一個(gè)pair,即(quick, the),通過(guò)quick預(yù)測(cè)the,假定num_noise=1,并且通過(guò)噪聲分布(一般就是詞的先驗(yàn)分布)采樣選出了噪聲詞sheep,當(dāng)前目標(biāo)變?yōu)椋?/p>

優(yōu)化的模型參數(shù)是詞向量(embedding vector),求損失函數(shù)的梯度,沿梯度方向更新這個(gè)參數(shù)。當(dāng)這個(gè)過(guò)程在整個(gè)數(shù)據(jù)集上不斷重復(fù)的時(shí)候,每個(gè)詞的詞向量就會(huì)不斷的“變來(lái)變?nèi)ァ?,直到模型能夠成功的從噪聲詞中區(qū)分真實(shí)詞。

我們可以通過(guò)投影到二維空間來(lái)可視化學(xué)習(xí)到的詞向量,用到了t-SNE降維技術(shù)。

實(shí)戰(zhàn)

有了前面的理論基礎(chǔ),實(shí)現(xiàn)代碼就比較容易了。

基礎(chǔ)實(shí)現(xiàn):tensorflow/examples/tutorials/word2vec/word2vec_basic.py

訓(xùn)練數(shù)據(jù)中,輸入是batch_size個(gè)target word id構(gòu)成的行向量,類標(biāo)是batch_size個(gè)context word id構(gòu)成的列向量:

# Input data.
train_inputs = tf.placeholder(tf.int32, shape=[batch_size])
train_labels = tf.placeholder(tf.int32, shape=[batch_size, 1])

詞向量通過(guò)均勻分布初始化,shape為詞表大小*詞向量維數(shù),通過(guò)tf.nn.embedding_lookup()查表將輸入轉(zhuǎn)為詞向量形式。

# Look up embeddings for inputs.
embeddings = tf.Variable(
    tf.random_uniform([vocabulary_size, embedding_size], -1.0, 1.0))
embed = tf.nn.embedding_lookup(embeddings, train_inputs)

噪聲對(duì)比估計(jì)的損失按照邏輯回歸模型定義,所以對(duì)于每一個(gè)詞,都要有對(duì)應(yīng)的權(quán)向量和偏置,通過(guò)截尾正態(tài)分布初始化(只保留兩個(gè)標(biāo)準(zhǔn)差以內(nèi)的值)。

# Construct the variables for the NCE loss
nce_weights = tf.Variable(
    tf.truncated_normal([vocabulary_size, embedding_size],
                        stddev=1.0 / math.sqrt(embedding_size)))
nce_biases = tf.Variable(tf.zeros([vocabulary_size]))

計(jì)算每個(gè)batch上的平均NCE損失

# Compute the average NCE loss for the batch.
# tf.nce_loss automatically draws a new sample of the negative labels each
# time we evaluate the loss.
loss = tf.reduce_mean(
  tf.nn.nce_loss(weights=nce_weights,
                 biases=nce_biases,
                 labels=train_labels,
                 inputs=embed,
                 num_sampled=num_sampled,
                 num_classes=vocabulary_size))

使用隨機(jī)梯度下降訓(xùn)練

# Construct the SGD optimizer using a learning rate of 1.0.
optimizer = tf.train.GradientDescentOptimizer(1.0).minimize(loss)

最后就是創(chuàng)建session進(jìn)行訓(xùn)練,詳細(xì)可看完整源代碼。

優(yōu)化

基礎(chǔ)代碼只是實(shí)現(xiàn)了skip-gram的負(fù)采樣模型,訓(xùn)練目標(biāo)為tf.nn.nce_loss(),還可以試試tf.nn.sampled_softmax_loss(),也可以自己定義。

另外,讀取數(shù)據(jù)也不是那么有效(單線程),可以試試New Data Formats中的方法自己定義數(shù)據(jù)reader,參考代碼:
word2vec.py.

如果還想進(jìn)一步提升效率,可以增加新的算子,參考代碼:word2vec_optimized.py

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 1.NLP當(dāng)前熱點(diǎn)方向 詞法/句法分析 詞嵌入(word embedding) 命名實(shí)體識(shí)別(Name Entit...
    __Aragorn閱讀 6,360評(píng)論 1 9
  • 前面的文章主要從理論的角度介紹了自然語(yǔ)言人機(jī)對(duì)話系統(tǒng)所可能涉及到的多個(gè)領(lǐng)域的經(jīng)典模型和基礎(chǔ)知識(shí)。這篇文章,甚至之后...
    我偏笑_NSNirvana閱讀 14,454評(píng)論 2 64
  • 在各種大舉深度學(xué)習(xí)大旗的公司中,Google公司無(wú)疑是旗舉得最高的,口號(hào)喊得最響亮的那一個(gè)。2013年末,Goog...
    chaaffff閱讀 17,230評(píng)論 0 29
  • 最近讀英文繪本比較少。 昨天中午我要求午休,但是呢,陳小冠突然發(fā)現(xiàn)這本書(shū),上午我看來(lái)著放在床頭了。先是自己翻看,然...
    木木sani閱讀 414評(píng)論 0 0
  • 吵架,焦慮,焦慮,吵架,一直翻來(lái)覆去,這么多年,我的狀態(tài)一直不好,所以一直沒(méi)有比較理想的生活確實(shí)和我的狀態(tài)有關(guān)。遇...
    kikin小鑫閱讀 179評(píng)論 0 0

友情鏈接更多精彩內(nèi)容