[机器学习与深度学习] - No.1 基于Negative Sampling SKip-Gram Word2vec模型学习总结

本文主要是介绍[机器学习与深度学习] - No.1 基于Negative Sampling SKip-Gram Word2vec模型学习总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基于Negative Sampling SKip-Gram Word2vec模型学习总结

1. Word2vec简介

Word2Vec是从大量文本语料中以无监督的方式学习语义知识的一种模型,它被大量地用在自然语言处理(NLP)中。那么它是如何帮助我们做自然语言处理呢?Word2Vec其实就是通过学习文本来用词向量的方式表征词的语义信息。Word2vec的结果是为了获得Word Embedding,我们又称为词嵌入。Word Embedding就是一个映射,将单词从原先所属的空间映射到新的多维空间中,也就是把原先词所在空间嵌入到一个新的空间中去。

Word2Vec模型实际上分为了两个部分,第一部分为建立模型,第二部分是通过模型获取嵌入词向量。基于训练数据构建一个神经网络,当这个模型训练好以后,我们并不会用这个训练好的模型处理新的任务,我们真正需要的是这个模型通过训练数据所学得的参数,例如隐层的权重矩阵。

如果之前了解过RNN 语言模型,会了解到,如果我们的词典中一共有10000个不同的词,那么我们每个词将会使用10000维的***one-hot***编码来表示。如果我们希望使用300维的特征向量代表每个词,我们设置模型的隐藏层节点为300维。输入层-隐藏层的权重矩阵为:10000X300维。那么权重矩阵的每一行即代表我们最后所需的word embedding

Word2vec模型结构:
这里写图片描述

输入层-隐藏层权重矩阵:
这里写图片描述

2. Skip-Gram

Word2Vec模型中,主要有Skip-Gram和CBOW两种模型,。Skip-Gram是给定中心词来预测上下文,CBOW是给定上下文,来预测中心词

这里写图片描述

如上图所示,蓝色是我们给定的中心词,上下两边绿色的字体表示我们中心词的上下文

这里写图片描述

1. 输入形式

Word2vec的输入形式为one-hot编码。假设从我们的训练文档中抽取出10000个唯一不重复的单词组成词汇表。我们对这10000个单词进行one-hot编码,得到的每个单词都是一个10000维的向量,向量每个维度的值只有0或者1,假如单词ants在词汇表中的出现位置为第3个,那么ants的向量就是一个第三维度取值为1,其他维都为0的10000维的向量(ants=[0, 0, 1, 0, …, 0])。

模型的输入如果为一个10000维的向量,那么输出也是一个10000维度(词汇表的大小)的向量,它包含了10000个概率,每一个概率代表着当前词是输入样本中output word的概率大小。

2. 样本形式

假设我们有一句话“The quick brown fox jumps over the lazy dog.”

  • 我们假设中间词为fox
  • 接下来,我们设置中心词fox的上下文范围。我们使用skip_window来表示我们从中心词的左右两侧选取的词的数量,num_skips 表示我们在中心词的上下文中选取作为输出词的数量。例如:当skip_window=2num_skips = 2,中心词为fox时 ,我们获得fox的上下文窗口为['quick','brown','jumps','over']。我们随机从窗口中选取两个词作为输出,那么我们的样本元组应该如同:('fox','brown')('fox','over')

这里写图片描述

如上图,列出了中心词所有的训练样本。在代码的实现过程中,我们会使用随机数来随机选取窗口中的输出词。

3. Negative Sampling

训练一个神经网络意味着要输入训练样本并且不断调整神经元的权重,从而不断提高对目标的准确预测。每当神经网络经过一个训练样本的训练,它的权重就会进行一次调整。语料词典的大小决定了我们的Skip-Gram神经网络将会拥有大规模的权重矩阵,所有的这些权重需要通过我们数以亿计的训练样本来进行调整,这是非常消耗计算资源的,并且实际中训练起来会非常慢。

1. 负采样简介

**负采样(negative sampling)**解决了这个问题,它是用来提高训练速度并且改善所得到词向量的质量的一种方法。不同于原本每个训练样本更新所有的权重,负采样每次让一个训练样本仅仅更新一小部分的权重,这样就会降低梯度下降过程中的计算量。

假如我们训练样本('fox','brown'),由于我们使用的是one-hot编码来表示词,我们期望对应“brown”单词的那个输出神经元输出1,其他剩余的所有输出神经元输出0,我们称所有输出为1的输出神经元对应的词为"positive"词,所有输出为0的神经元对应的词为"negative"词

使用负采样的方法,我们不对所有输出神经元对应的权值进行更新,只是随机选取几个"negative"词,更新他们对应的权重。当然,我们也会更新"positive"的对应权值。

如上面所述,加入我们有10000个单词,每个单词用300维表示,那么我们的权重矩阵为***10000x300***维的矩阵。我们每次更新需要更新3000000个值。如果我们只更新随机选取的5个"negative"单词和一个"positive"的权重,那么我们只需要更新1800个值,相当于之前0.06%的计算量。

2. 负采样点选取

本质上来说,一个单词备选做为"negative word"的概率和他出现的频率有关,出现频次越高的单词越容易备选做"negative word"。这就是我们对采样过程的一个大致要求,本质上是一个带权采样的问题。

我们使用一个比较通俗的描述来解释一下带权采样:

设词典D中的每个词 ω \omega ω 对应一个线段 l ( ω ) l(\omega) l(ω) ,长度为:
l e n ( ω ) = c o u n t e r ( ω ) ∑ u ∈ D c o u n t e r ( u ) len(\omega) = \frac{counter(\omega)}{\sum_{u\in D} counter(u)} len(ω)=uDcounter(u)counter(ω)
这里的counter()表示一个词在语料中出现的次数。现在将这些线段首位连接在一起,形成一个长度为1的单位线段,如果在线段上随机的打点,那么长度长(频率大)的线段被打中的概率就大

所以,根据 { l e n ( ) j } j = 0 N \{len()_j\}_{j=0}^N {len()j}j=0N 可以得到区间[0,1]上的一个非等距剖分,共有N个剖分区间

接着,我们引入区间[0,1]上的一个等距离剖分,剖分解点为 { m j } j = 0 M \{m_j\}^M_{j=0} {mj}j=0M ,其中M >> N, 如下图所示
这里写图片描述
如上图所示,采样就简单了。我们生成一个[1,M-1]之间的随机整数r,然后查看该整数r落在了哪个词对应的线段内,那么该单词就是采样点。如果碰巧遇到了’'positive word",那么就跳过重新选取。

在word2vec的C语言实现中,使用了下面的公式来计算单词被选做负样本的概率。每个单词被选为“negative words”的概率计算公式与其出现的频次有关。
P ( ω j ) = f ( ω j ) 3 4 ∑ j = 0 n ( f ( ω j ) 3 4 ) P(\omega_j) = \frac{{f(\omega_j)}^{\frac{3}{4}}}{\sum^n_{j=0} {(f(\omega_j)^{\frac{3}{4}})}} P(ωj)=j=0n(f(ωj)43)f(ωj)43
其中 f ( ω j ) f(\omega_j) f(ωj) 代表词 ω j {\omega_j} ωj 的在整个语料中出现的频次

####4. 参考文章:

http://blog.csdn.net/itplus/article/details/37998797

https://www.leiphone.com/news/201706/eV8j3Nu8SMqGBnQB.html

http://www.thushv.com/natural_language_processing/word2vec-part-1-nlp-with-deep-learning-with-tensorflow-skip-gram/

http://mccormickml.com/2017/01/11/word2vec-tutorial-part-2-negative-sampling/

这篇关于[机器学习与深度学习] - No.1 基于Negative Sampling SKip-Gram Word2vec模型学习总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/897396

相关文章

Python中logging模块用法示例总结

《Python中logging模块用法示例总结》在Python中logging模块是一个强大的日志记录工具,它允许用户将程序运行期间产生的日志信息输出到控制台或者写入到文件中,:本文主要介绍Pyt... 目录前言一. 基本使用1. 五种日志等级2.  设置报告等级3. 自定义格式4. C语言风格的格式化方法

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3

Spring 依赖注入与循环依赖总结

《Spring依赖注入与循环依赖总结》这篇文章给大家介绍Spring依赖注入与循环依赖总结篇,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Spring 三级缓存解决循环依赖1. 创建UserService原始对象2. 将原始对象包装成工

Java MCP 的鉴权深度解析

《JavaMCP的鉴权深度解析》文章介绍JavaMCP鉴权的实现方式,指出客户端可通过queryString、header或env传递鉴权信息,服务器端支持工具单独鉴权、过滤器集中鉴权及启动时鉴权... 目录一、MCP Client 侧(负责传递,比较简单)(1)常见的 mcpServers json 配置

Maven中生命周期深度解析与实战指南

《Maven中生命周期深度解析与实战指南》这篇文章主要为大家详细介绍了Maven生命周期实战指南,包含核心概念、阶段详解、SpringBoot特化场景及企业级实践建议,希望对大家有一定的帮助... 目录一、Maven 生命周期哲学二、default生命周期核心阶段详解(高频使用)三、clean生命周期核心阶

深度剖析SpringBoot日志性能提升的原因与解决

《深度剖析SpringBoot日志性能提升的原因与解决》日志记录本该是辅助工具,却为何成了性能瓶颈,SpringBoot如何用代码彻底破解日志导致的高延迟问题,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言第一章:日志性能陷阱的底层原理1.1 日志级别的“双刃剑”效应1.2 同步日志的“吞吐量杀手”

MySQL中查询和展示LONGBLOB类型数据的技巧总结

《MySQL中查询和展示LONGBLOB类型数据的技巧总结》在MySQL中LONGBLOB是一种二进制大对象(BLOB)数据类型,用于存储大量的二进制数据,:本文主要介绍MySQL中查询和展示LO... 目录前言1. 查询 LONGBLOB 数据的大小2. 查询并展示 LONGBLOB 数据2.1 转换为十

Unity新手入门学习殿堂级知识详细讲解(图文)

《Unity新手入门学习殿堂级知识详细讲解(图文)》Unity是一款跨平台游戏引擎,支持2D/3D及VR/AR开发,核心功能模块包括图形、音频、物理等,通过可视化编辑器与脚本扩展实现开发,项目结构含A... 目录入门概述什么是 UnityUnity引擎基础认知编辑器核心操作Unity 编辑器项目模式分类工程

深度解析Python yfinance的核心功能和高级用法

《深度解析Pythonyfinance的核心功能和高级用法》yfinance是一个功能强大且易于使用的Python库,用于从YahooFinance获取金融数据,本教程将深入探讨yfinance的核... 目录yfinance 深度解析教程 (python)1. 简介与安装1.1 什么是 yfinance?