[学习笔记] [机器学习] 12. [上] HMM 隐马尔可夫算法(马尔科夫链、HMM 三类问题、前后后向算法、维特比算法、鲍姆-韦尔奇算法、API 及实例)

本文主要是介绍[学习笔记] [机器学习] 12. [上] HMM 隐马尔可夫算法(马尔科夫链、HMM 三类问题、前后后向算法、维特比算法、鲍姆-韦尔奇算法、API 及实例),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

学习目标

  • 了解什么是马尔科夫链
  • 知道什么是 HMM 模型
  • 知道前向后向算法评估观察序列概率
  • 知道维特比算法解码隐藏状态序列
  • 了解鲍姆-韦尔奇算法
  • 知道 HMM 模型 API 的使用

1. 马尔科夫链

学习目标

  • 知道什么是马尔科夫链

在机器学习算法中,马尔可夫链(Markov Chain)是个很重要的概念。马尔可夫链,又称离散时间马尔可夫链(Discrete-Time Markov Chain),因俄国数学家安德烈-马尔可夫得名。

1.1 马尔科夫链的简介

马尔科夫链即为状态空间中从一个状态到另一个状态转换的随机过程。

在这里插入图片描述

我们看图中的灰色圆点,它可以自身到自身(A → A / B → B),也可以自身到下一点(A → B / B → A),与前几步是怎么走的没有关系。

该过程要求具备“无记忆”的性质:即下一状态的概率分布只能由当前状态决定,在时间序列中它前面的事件均与之无关。这种特定类型的“无记忆性”称作马尔可夫性质。

马尔科夫链作为实际过程的统计模型具有许多应用,在机器学习和人工智能的领域中有着广泛的应用,如强化学习、自然语言处理、金融领域、天气预测和语音识别等方面。

在马尔可夫链的每一步,系统根据概率分布,可以从一个状态变到另一个状态,也可以保持当前状态。

  • 状态的改变叫做转移
  • 与不同的状态改变相关的概率叫做转移概率

马尔可夫链的数学表示为:

P ( x t + 1 ∣ ⋅ ⋅ ⋅ , x t − 2 , x t − 1 , x t ) = P ( x t + 1 ∣ x t ) P(x_{t+1}|···, x_{t-2}, x_{t-1}, x_t) = P(x_{t+1}|x_t) P(xt+1⋅⋅⋅,xt2,xt1,xt)=P(xt+1xt)

上面这个公式说明,马尔科夫链的转移概率与 t t t 之前的概率无关,只与 t t t 有关

既然某一时刻状态转移的概率只依赖前一个状态,那么只要求出系统中任意两个状态之间的转移概率,这个马尔科夫链的模型就定了

1.2 马尔科夫链的经典举例

下图中的马尔科夫链是用来表示股市模型,共有三种状态:牛市(Bull market)、熊市(Bear market)和横盘(Stagnant market)。每一个状态都以一定的概率转化到下一个状态。比如,牛市以0.025的概率转化到横盘的状态。

下图中的马尔科夫链是用来表示股市模型,共有三种状态:

  • 牛市(Bull market):股票的持续上涨,当涨幅超过 20% 的时候,就被叫做牛市
  • 熊市(Bear market):股票的持续下跌,当下跌幅度超过 20% 的时候,就被称为熊市
  • 横盘(Stagnant market):股票价格在一段时间内基本保持不变,没有明显的涨跌趋势

每一个状态都以一定的概率转化到下一个状态。比如,牛市以 0.025 的概率转化到横盘的状态。

在这里插入图片描述

这个状态概率转化图可以以矩阵的形式表示


如果我们定义矩阵 P P P 某一位置 P ( i , j ) P(i, j) P(i,j) 的值为 P ( j ∣ i ) P(j | i) P(ji),即从状态 i i i 变为状态 j j j 的概率。另外定义牛市、熊市、横盘的状态分别为 0、1、2,这样我们得到了马尔科夫链模型的状态转移矩阵:

P = ( 0.9 0.075 0.025 0.15 0.8 0.05 0.25 0.25 0.5 ) P = \begin{pmatrix} 0.9 & 0.075 & 0.025\\ 0.15 & 0.8 & 0.05\\ 0.25 & 0.25 & 0.5 \end{pmatrix} P= 0.90.150.250.0750.80.250.0250.050.5

上面这个矩阵:

  • 第一行表示牛市;第二行表示熊市;第三行表示横盘
  • 第一列表示牛市;第二列表示熊市;第三列表示横盘
  • 矩阵的对角线表示转移到自身

当这个状态转移矩阵 P P P 确定以后,整个股市模型就已经确定。


小结

  • 马尔科夫链即为:状态空间中从一个状态到另一个状态转换的随机过程。
    • 该过程要求具备“无记忆”的性质:下一状态的概率分布只能由当前状态决定,在时间序列中它前面的事件均与之无关

2. HMM 简介

学习目标

  • 通过举例了解什么是 HMM 模型

隐马尔可夫模型(Hidden Markov Model,HMM)是统计模型,它用来描述一个含有隐含未知参数的马尔可夫过程。它是关于时序的概率模型,描述由一个隐藏的马尔可夫链随机生成不可观测的状态随机序列,再由各个状态生成一个观测从而产生观测随机序列的过程。

其难点是从可观察的参数中确定该过程的隐含参数,然后利用这些参数来作进一步的分析

隐马尔可夫模型(HMM)在语音识别、机器翻译、中文分词、命名实体识别、词性标注、基因识别等领域有广泛的使用。它也被应用于手势、字体识别,地平面估计,网络安全中的入侵检测,生物中的DNA分段序列的预测以及故障的预测和诊断。

2.1 HMM 的简单案例

下面我们一起用一个简单的例子来对 HMM 进行阐述:假设我们手里有三个不同的骰子。

  • 第一个骰子是我们平常见的骰子(称这个骰子为 D6),它有 6 个面,每个面 (1, 2, 3, 4, 5, 6) 出现的概率都是 1 6 \frac{1}{6} 61
  • 第二个骰子是个四面体(称这个骰子为 D4),每个面 (1, 2, 3, 4) 出现的概率都是 1 4 \frac{1}{4} 41
  • 第三个骰子有八个面(称这个骰子为 D8),每个面 (1, 2, 3, 4, 5, 6, 7, 8) 出现的概率都是 1 8 \frac{1}{8} 81

在这里插入图片描述

我们开始掷骰子,我们先从三个骰子里挑一个,挑到每一个骰子的概率都是 1 3 \frac{1}{3} 31。然后我们掷骰子,得到一个数字(1, 2, 3, 4, 5, 6, 7, 8 中的一个)。不停的重复上述过程,我们会得到一串数字,每个数字都是 1, 2, 3, 4, 5, 6, 7, 8 中的一个。


例如,我们可能得到这么一串数字(掷骰子 10 次):1 6 3 5 2 7 3 5 2 4。我们称这串数字叫做 可见状态链

但是在隐马尔可夫模型中,我们不仅仅有这么一串可见状态链,还有一串隐含状态链。在这个例子里,这串隐含状态链就是你用的骰子的序列

在这里插入图片描述

在这里插入图片描述

搞清楚哪些是隐含状态,哪些是可见状态


比如,隐含状态链有可能是:D6 D8 D8 D6 D4 D8 D6 D6 D4 D8。

一般来说,HMM 中说到的马尔可夫链其实是指隐含状态链,因为隐含状态(骰子)之间存在转换概率(Transition Probability)。

在我们这个例子里,D6 的下一个状态是 D4 或 D6 或 D8 的概率都是 1 3 \frac{1}{3} 31。D4 和 D8 的下一个状态是 D4 或 D6 或 D8 的转换概率也都一样是 1 3 \frac{1}{3} 31

这样设定是为了最开始容易说清楚,但是我们其实是可以随意设定转换概率的。比如,我们可以这样定义,D6 后面不能接 D4,D6 后面是 D6 的概率是 0.9,是 D8 的概率是 0.1。这样就是一个新的 HMM。

同样的,尽管可见状态之间没有转换概率,但是隐含状态和可见状态之间有一个概率叫做输出概率(Emission Probability)


对于我们的例子来说,六面骰(D6)投出 1 的输出概率是 1 6 \frac{1}{6} 61。投出 2, 3, 4, 5, 6 的概率也都是 1 6 \frac{1}{6} 61。我们同样可以对输出概率进行其他定义。比如,我有一个被赌场动过手脚的六面骰子,掷出来是 1 的概率更大,是 1 2 \frac{1}{2} 21,掷出来是 2, 3, 4, 5, 6 的概率是 1 10 \frac{1}{10} 101

其实对于 HMM 来说,如果提前知道所有隐含状态之间的转换概率和所有隐含状态到所有可见状态之间的输出概率,做模拟是相当容易的。但是在应用 HMM 模型时候呢,往往是缺失了一部分信息的。

  • 有时候你知道骰子有几种,每种骰子是什么,但是不知道掷出来的骰子序列
  • 有时候你只是看到了很多次掷骰子的结果,剩下的什么都不知道

如果应用算法去估计这些缺失的信息,就成了一个很重要的问题。这些问题我们会在后面详细讲。

2.2 案例进阶

2.2.1 问题阐述

和 HMM 模型相关的算法主要分为三类(分别解决三种问题)。

2.2.1.1 第一类算法

知道骰子有几种(隐含状态数量),每种骰子是什么(转换概率),根据掷骰子掷出的结果(可见状态链),我们想知道每次掷出来的都是哪种骰子(隐含状态链)。

即:根据可见状态链 + 转换概率 → 隐含状态链

这些结果在投掷的时候用的哪些骰子

注意

  • 隐含状态数量:有几种骰子(本例中有 3 种骰子)
  • 转换概率:每种骰子是什么(本例中为 D4 D6 D8)
  • 可见状态链:骰子掷出的结果(本例中为 1 ~ 8)

对于这个问题,在语音识别领域叫做解码问题。

这个问题其实有两种解法,会给出两个不同的答案。每个答案都对,只不过这些答案的意义不一样。

  • 第一种解法:求最大似然状态路径。说通俗点就是我们求一串骰子序列(比如 D6 D4 D8 D6 D6 D4 …),这串骰子序列产生观测结果的概率最大(比如 0 1 0 2 2 3 4 …)。

  • 第二种解法:不再是求一组骰子序列了,而是求每次掷出的骰子分别是某种骰子的概率。比如说我们看到结果后,我可以求得第一次掷骰子是 D4 的概率是 0.5,D6 的概率是 0.3,D8 的概率是 0.2。

2.2.1.2 第二类算法

还是知道骰子有几种(隐含状态数量),每种骰子是什么(转换概率),根据掷骰子掷出的结果(可见状态链),我想知道掷出这个结果的概率(下次再投出这个结果的概率有多大)。

看似这个问题意义不大,因为你掷出来的结果很多时候都对应了一个比较大的概率。问这个问题的目的呢,其实是检测观察到的结果和已知的模型是否吻合

如果很多次结果都对应了比较小的概率,那么就说明我们已知的模型很有可能是错的,有人偷偷把我们的骰子给换了。

2.2.1.3 第三类算法

知道骰子有几种(隐含状态数量),不知道每种骰子是什么(转换概率),观测到很多次掷骰子的结果(可见状态链),我想反推出每种骰子是什么(转换概率)。

即:根据 隐含状态数量 + 可见状态链 → 转换概率

这里我们主要想知道骰子之间的转换概率是多少,和我们的预期是否一致!

这个问题很重要,因为这是最常见的情况

很多时候我们只有可见结果,不知道 HMM 模型里的参数,我们需要从可见结果估计出这些参数,这是建模的一个必要步骤。

2.2.2 问题解决

2.2.2.1 一个简单问题【对应问题二】

其实这个问题实用价值不高。由于对下面较难的问题有帮助,所以先在这里提一下。

知道骰子有几种(隐含状态数量)、每种骰子是什么(转换概率),根据掷骰子掷出的结果(可见状态链),我们想知道掷出这个结果的概率(下次再投出这个结果的概率有多大)。

在这里插入图片描述

解法无非就是概率相乘:

P = P ( D 6 ) × P ( D 6 → 1 ) × P ( D 6 → D 8 ) × P ( D 8 → 6 ) × P ( D 8 → D 8 ) × P ( D 8 → 3 ) = 1 3 × 1 6 × 1 3 × 1 8 × 1 3 × 1 8 = 0.00005787 \begin{aligned} P & = P(D6) \times P(D6 \rightarrow 1) \times P(D6 \rightarrow D8) \times P(D8 \rightarrow 6) \times P(D8 \rightarrow D8) \times P(D8 \rightarrow 3)\\ & = \frac{1}{3} \times \frac{1}{6} \times \frac{1}{3} \times \frac{1}{8} \times \frac{1}{3} \times \frac{1}{8}\\ & = 0.00005787 \end{aligned} P=P(D6)×P(D61)×P(D6D8)×P(D86)×P(D8D8)×P(D83)=31×61×31×81×31×81=0.00005787

2.2.2.2 看见不可见的,破解骰子序列【对应问题一】

问题一:知道骰子有几种(隐含状态数量),每种骰子是什么(转换概率),根据掷骰子掷出的结果(可见状态链),我们想知道每次掷出来的都是哪种骰子(隐含状态链)。

即:根据可见状态链 + 转换概率 → 隐含状态链

这里我们使用第一种解法,即最大似然路径问题。

举例来说,我知道我有三个骰子,六面骰 D6,四面骰 D4,八面骰 D8。我也知道我掷了十次的结果(1 6 3 5 2 7 3 5 2 4)。但我们不知道每次用了那种骰子,我们想知道最有可能的骰子序列(这些结果在投掷的时候用的哪些骰子)。

其实最简单粗暴的方法就是穷举所有可能的骰子序列,然后依照上一个问题的解法把每个序列对应的概率算出来。然后我们从里面把对应最大概率的序列挑出来就行了。

如果马尔可夫链不长,这样做当然可行。但如果马尔科夫链很长的话,穷举的数量太大,就很难完成了。

另外一种很有名的算法叫做维特比算法(Viterbi algorithm)。要理解这个算法,我们先看几个简单的例子。

首先,如果我们只掷一次骰子:

在这里插入图片描述

看到结果为 1,对应的最大概率骰子序列就是 D4。因为 D4 产生 1 的概率是 1 4 \frac{1}{4} 41,高于 1 6 \frac{1}{6} 61 1 8 \frac{1}{8} 81

把这个情况拓展,我们掷两次骰子:

在这里插入图片描述

结果为 1 6。此时问题变得复杂起来,我们需要计算三个值,分别是第二个骰子是 D6 或 D4 或 D8 的最大概率。显然,要取到最大概率,第一个骰子必须为 D4。这时,第二个骰子取到 D6 的最大概率是:

P 2 ( D 6 ) = P ( D 4 ) × P ( D 4 → 1 ) ‾ 第一次投掷 × P ( D 4 → D 6 ) × P ( D 6 → 6 ) ‾ 第二次投掷 = 1 3 × 1 4 × 1 3 × 1 6 \begin{aligned} P2(D6) &= \underset{第一次投掷}{\underline{P(D4) \times P(D4 \rightarrow 1)}} \times \underset{第二次投掷}{\underline{P(D4 \rightarrow D6) \times P(D6 \rightarrow 6)}}\\ & = \frac{1}{3} \times \frac{1}{4} \times \frac{1}{3} \times \frac{1}{6} \end{aligned} P2(D6)=第一次投掷P(D4)×P(D41)×第二次投掷P(D4D6)×P(D66)=31×41×31×61

同样的,我们可以计算第二个骰子是 D4 或 D8 时的最大概率。我们发现,第二个骰子取到 D6 的概率最大。而使这个概率最大时,第一个骰子为 D4 。所以最大概率骰子序列就是 D4 D6。继续拓展,我们掷三次骰子:

在这里插入图片描述

同样,我们计算第三个骰子分别是 D6 D4 D8的最大概率。我们再次发现,要取到最大概率,第二个骰子必须为 D6。这时,第三个骰子取到 D4 的最大概率是:

P 3 ( D 4 ) = P ( D 4 ) × P ( D 4 → 1 ) ‾ 第一次投掷 × P ( D 4 → D 6 ) × P ( D 6 → 6 ) ‾ 第二次投掷 × P ( D 6 → D 4 ) × P ( D 4 → 3 ) ‾ 第三次投掷 = 1 3 × 1 4 × 1 3 × 1 6 × 1 3 × 1 4 \begin{aligned} P3(D4) &= \underset{第一次投掷}{\underline{P(D4) \times P(D4 \rightarrow 1)}} \times \underset{第二次投掷}{\underline{P(D4 \rightarrow D6) \times P(D6 \rightarrow 6)}} \times \underset{第三次投掷}{\underline{P(D6 \rightarrow D4) \times P(D4 \rightarrow 3)}}\\ & = \frac{1}{3} \times \frac{1}{4} \times \frac{1}{3} \times \frac{1}{6} \times \frac{1}{3} \times \frac{1}{4} \end{aligned} P3(D4)=第一次投掷P(D4)×P(D41)×第二次投掷P(D4D6)×P(D66)×第三次投掷P(D6D4)×P(D43)=31×41×31×61×31×41

同上,我们可以计算第三个骰子是 D6 或 D8 时的最大概率。我们发现,第三个骰子取到 D4 的概率最大。而使这个概率最大时,第二个骰子为 D6,第一个骰子为 D4。所以最大概率骰子序列就是 D4 D6 D4。


到这里,我们应该看出点规律了。既然掷骰子一、二、三次可以算,掷多少次都可以以此类推。

我们发现,我们要求最大概率骰子序列时要做这么几件事情:

  • 首先,不管序列多长,要从序列长度为 1 算起,算序列长度为 1 时取到每个骰子的最大概率。
  • 然后,逐渐增加长度,每增加一次长度,重新算一遍在这个长度下最后一个位置取到每个骰子的最大概率。因为上一个长度下的取到每个骰子的最大概率都算过了、重新计算的话其实不难(类似于递归)。
  • 当我们算到最后一位时,就知道最后一位是哪个骰子的概率最大了。然后,我们要把对应这个最大概率的序列从后往前推出来(回溯)

维特比算法借鉴了动态规划的思想。

2.2.2.3 谁动了我的骰子?【对应问题三】

问题3:知道骰子有几种(隐含状态数量),不知道每种骰子是什么(转换概率),观测到很多次掷骰子的结果(可见状态链),我想反推出每种骰子是什么(转换概率)。

这里我们主要想知道骰子之间的转换概率是多少,和我们的预期是否一致!

即:根据 隐含状态数量 + 可见状态链 → 转换概率

这个问题很重要,因为这是最常见的情况

很多时候我们只有可见结果,不知道 HMM 模型里的参数,我们需要从可见结果估计出这些参数,这是建模的一个必要步骤。


比如说你怀疑自己的六面骰 D6 被赌场动过手脚了,有可能被换成另一种六面骰 D ^ 6 \hat{D}6 D^6,这种六面骰掷出来是 1 的概率更大( 1 2 \frac{1}{2} 21),掷出来是 2, 3, 4, 5, 6 的概率是 1 10 \frac{1}{10} 101。此时我们该怎么办?

其实答案很简单,算一算正常的三个骰子掷出一段序列的概率,再算一算不正常的六面骰和另外两个正常骰子掷出这段序列的概率。如果前者比后者小,我们就要小心了。

比如说掷骰子的结果是:

在这里插入图片描述

要算用正常的三个骰子掷出这个结果的概率,其实就是将所有可能情况的概率进行加和计算。

同样,简单粗暴的方法就是把穷举所有的骰子序列,还是计算每个骰子序列对应的概率。但是这次我们不挑最大值了,而是把所有算出来的概率相加,得到的总概率就是我们要求的结果

这个方法依然不能应用于太长的骰子序列(马尔可夫链)。我们会应用一个 和前一个问题 类似的解法,只不过前一个问题关心的是概率最大值,这个问题关心的是概率之和。解决这个问题的算法叫做 前向算法(Forward algorithm)

首先,如果我们只掷一次骰子:

在这里插入图片描述

看到结果为 1。产生这个结果的总概率可以按照如下计算,总概率为 0.18:

P 1 P1 P1 P 2 P2 P2 P 3 P3 P3
D6 1 3 × 1 6 \frac{1}{3} \times \frac{1}{6} 31×61
D4 1 3 × 1 4 \frac{1}{3} \times \frac{1}{4} 31×41
D8 1 3 × 1 8 \frac{1}{3} \times \frac{1}{8} 31×81
总计 0.18 0.18 0.18

把这个情况拓展,我们掷两次骰子:

在这里插入图片描述

看到结果为 1 6。产生这个结果的总概率可以按照如下计算,总概率为 0.05:

P 1 P1 P1 P 2 P2 P2 P 3 P3 P3
D6 1 3 + 1 6 \frac{1}{3} + \frac{1}{6} 31+61 P 1 ( D 6 ) × 1 3 × 1 6 ‾ 第一次 D 6 , 第二次 D 6 + P 1 ( D 4 ) × 1 3 × 1 6 ‾ 第一次 D 4 , 第二次 D 6 + P 1 ( D 8 ) × 1 3 × 1 6 ‾ 第一次 D 8 , 第二次 D 6 \underset{第一次D6,第二次D6}{\underline{P1(D6) \times \frac{1}{3}\times \frac{1}{6}}} + \underset{第一次D4,第二次D6}{\underline{P1(D4) \times \frac{1}{3} \times \frac{1}{6}}} + \underset{第一次D8, 第二次D6}{\underline{P1(D8) \times \frac{1}{3} \times \frac{1}{6}}} 第一次D6,第二次D6P1(D6)×31×61+第一次D4,第二次D6P1(D4)×31×61+第一次D8,第二次D6P1(D8)×31×61
D4 1 3 + 1 4 \frac{1}{3} + \frac{1}{4} 31+41 P 1 ( D 6 ) × 1 3 × 0 ‾ 第一次 D 6 , 第二次 D 4 + P 1 ( D 4 ) × 1 3 × 0 ‾ 第一次 D 4 , 第二次 D 4 + P 1 ( D 8 ) × 1 3 × 0 ‾ 第一次 D 8 , 第二次 D 4 \underset{第一次D6,第二次D4}{\underline{P1(D6) \times \frac{1}{3}\times 0}} + \underset{第一次D4,第二次D4}{\underline{P1(D4) \times \frac{1}{3} \times 0}} + \underset{第一次D8, 第二次D4}{\underline{P1(D8) \times \frac{1}{3} \times 0}} 第一次D6,第二次D4P1(D6)×31×0+第一次D4,第二次D4P1(D4)×31×0+第一次D8,第二次D4P1(D8)×31×0
D8 1 3 + 1 8 \frac{1}{3} + \frac{1}{8} 31+81 P 1 ( D 6 ) × 1 3 × 1 8 ‾ 第一次 D 6 , 第二次 D 8 + P 1 ( D 4 ) × 1 3 × 1 8 ‾ 第一次 D 4 , 第二次 D 8 + P 1 ( D 8 ) × 1 3 × 1 8 ‾ 第一次 D 8 , 第二次 D 8 \underset{第一次D6,第二次D8}{\underline{P1(D6) \times \frac{1}{3}\times \frac{1}{8}}} + \underset{第一次D4,第二次D8}{\underline{P1(D4) \times \frac{1}{3} \times \frac{1}{8}}} + \underset{第一次D8, 第二次D8}{\underline{P1(D8) \times \frac{1}{3} \times \frac{1}{8}}} 第一次D6,第二次D8P1(D6)×31×81+第一次D4,第二次D8P1(D4)×31×81+第一次D8,第二次D8P1(D8)×31×81
总计 0.18 0.18 0.18(第一次为 1 的概率) 0.05 0.05 0.05(第一次为 1,第二次为 6 的概率)

对于 D4 而言,是投掷不出来 6 的,因此概率为 0。

继续拓展,我们掷三次骰子:

在这里插入图片描述

看到结果为 1 6 3。产生这个结果的总概率可以按照如下计算,总概率为 0.03:

P 1 P1 P1 P 2 P2 P2 P 3 P3 P3
D6 1 3 + 1 6 \frac{1}{3} + \frac{1}{6} 31+61 P 1 ( D 6 ) × 1 3 × 1 6 ‾ 第一次 D 6 , 第二次 D 6 + P 1 ( D 4 ) × 1 3 × 1 6 ‾ 第一次 D 4 , 第二次 D 6 + P 1 ( D 8 ) × 1 3 × 1 6 ‾ 第一次 D 8 , 第二次 D 6 \underset{第一次D6,第二次D6}{\underline{P1(D6) \times \frac{1}{3}\times \frac{1}{6}}} + \underset{第一次D4,第二次D6}{\underline{P1(D4) \times \frac{1}{3} \times \frac{1}{6}}} + \underset{第一次D8, 第二次D6}{\underline{P1(D8) \times \frac{1}{3} \times \frac{1}{6}}} 第一次D6,第二次D6P1(D6)×31×61+第一次D4,第二次D6P1(D4)×31×61+第一次D8,第二次D6P1(D8)×31×61 P 2 ( D 6 ) × 1 3 × 1 6 ‾ 第二次 D 6 , 第三次 D 6 + P 2 ( D 4 ) × 1 3 × 1 6 ‾ 第二次 D 4 , 第三次 D 6 + P 2 ( D 8 ) × 1 3 × 1 6 ‾ 第二次 D 8 , 第三次 D 6 \underset{第二次D6,第三次D6}{\underline{P2(D6) \times \frac{1}{3} \times \frac{1}{6}}} + \underset{第二次D4,第三次D6}{\underline{P2(D4) \times \frac{1}{3} \times \frac{1}{6}}} + \underset{第二次D8,第三次D6}{\underline{P2(D8) \times \frac{1}{3} \times \frac{1}{6}}} 第二次D6,第三次D6P2(D6)×31×61+第二次D4,第三次D6P2(D4)×31×61+第二次D8,第三次D6P2(D8)×31×61
D4 1 3 + 1 4 \frac{1}{3} + \frac{1}{4} 31+41 P 1 ( D 6 ) × 1 3 × 0 ‾ 第一次 D 6 , 第二次 D 4 + P 1 ( D 4 ) × 1 3 × 0 ‾ 第一次 D 4 , 第二次 D 4 + P 1 ( D 8 ) × 1 3 × 0 ‾ 第一次 D 8 , 第二次 D 4 \underset{第一次D6,第二次D4}{\underline{P1(D6) \times \frac{1}{3}\times 0}} + \underset{第一次D4,第二次D4}{\underline{P1(D4) \times \frac{1}{3} \times 0}} + \underset{第一次D8, 第二次D4}{\underline{P1(D8) \times \frac{1}{3} \times 0}} 第一次D6,第二次D4P1(D6)×31×0+第一次D4,第二次D4P1(D4)×31×0+第一次D8,第二次D4P1(D8)×31×0 P 2 ( D 6 ) × 1 3 × 1 4 ‾ 第二次 D 6 , 第三次 D 4 + P 2 ( D 4 ) × 1 3 × 1 4 ‾ 第二次 D 4 , 第三次 D 4 + P 2 ( D 8 ) × 1 3 × 1 4 ‾ 第二次 D 8 , 第三次 D 4 \underset{第二次D6,第三次D4}{\underline{P2(D6) \times \frac{1}{3} \times \frac{1}{4}}} + \underset{第二次D4,第三次D4}{\underline{P2(D4) \times \frac{1}{3} \times \frac{1}{4}}} + \underset{第二次D8,第三次D4}{\underline{P2(D8) \times \frac{1}{3} \times \frac{1}{4}}} 第二次D6,第三次D4P2(D6)×31×41+第二次D4,第三次D4P2(D4)×31×41+第二次D8,第三次D4P2(D8)×31×41
D8 1 3 + 1 8 \frac{1}{3} + \frac{1}{8} 31+81 P 1 ( D 6 ) × 1 3 × 1 8 ‾ 第一次 D 6 , 第二次 D 8 + P 1 ( D 4 ) × 1 3 × 1 8 ‾ 第一次 D 4 , 第二次 D 8 + P 1 ( D 8 ) × 1 3 × 1 8 ‾ 第一次 D 8 , 第二次 D 8 \underset{第一次D6,第二次D8}{\underline{P1(D6) \times \frac{1}{3}\times \frac{1}{8}}} + \underset{第一次D4,第二次D8}{\underline{P1(D4) \times \frac{1}{3} \times \frac{1}{8}}} + \underset{第一次D8, 第二次D8}{\underline{P1(D8) \times \frac{1}{3} \times \frac{1}{8}}} 第一次D6,第二次D8P1(D6)×31×81+第一次D4,第二次D8P1(D4)×31×81+第一次D8,第二次D8P1(D8)×31×81 P 2 ( D 6 ) × 1 3 × 1 8 ‾ 第二次 D 6 , 第三次 D 8 + P 2 ( D 4 ) × 1 3 × 1 8 ‾ 第二次 D 4 , 第三次 D 8 + P 2 ( D 8 ) × 1 3 × 1 8 ‾ 第二次 D 8 , 第三次 D 8 \underset{第二次D6,第三次D8}{\underline{P2(D6) \times \frac{1}{3} \times \frac{1}{8}}} + \underset{第二次D4,第三次D8}{\underline{P2(D4) \times \frac{1}{3} \times \frac{1}{8}}} + \underset{第二次D8,第三次D8}{\underline{P2(D8) \times \frac{1}{3} \times \frac{1}{8}}} 第二次D6,第三次D8P2(D6)×31×81+第二次D4,第三次D8P2(D4)×31×81+第二次D8,第三次D8P2(D8)×31×81
总计 0.18 0.18 0.18(第一次为 1 的概率) 0.05 0.05 0.05(第一次为 1,第二次为 6 的概率) 0.03 0.03 0.03(第一次为 1,第二次为 6,第三次为 3 的概率)

同样的,我们一步一步的算,有多长算多长,再长的马尔可夫链总能算出来的。

用同样的方法,也可以算出不正常的六面骰和另外两个正常骰子掷出这段序列的概率,然后我们比较一下这两个概率大小,就能知道你的骰子是不是被人换了


小结

  • 隐马尔可夫模型(Hidden Markov Model,HMM)是统计模型,它用来描述一个含有隐含未知参数的马尔可夫过程。
  • 常见术语:
    • 可见状态链
    • 隐含状态链
    • 转换概率
    • 输出概率

3. HMM 模型基础

学习目标

  • 了解 HMM 模型解决的问题的主要特征
  • 知道 HMM 模型的两个重要假设
  • 指导 HMM 观测序列的生成过程
  • 知道 HMM 模型的三个基本问题

3.1 什么样的问题需要 HMM 模型

首先我们来看看什么样的问题可以用 HMM 模型解决。使用 HMM 模型时我们的问题一般有这两个特征:

  1. 我们的问题是基于序列的,比如时间序列,或者状态序列
  2. 我们的问题中有两类数据:
    1. 一类序列数据是可以观测到的,即观测序列
    2. 另一类数据是不能观察到的,即隐藏状态序列,简称状态序列

有了这两个特征,那么这个问题一般可以用 HMM 模型来尝试解决。


这样的问题在实际生活中是很多的。比如:现在写博客,在键盘上敲出来的一系列字符就是观测序列,而实际想写的一段话就是隐藏状态序列。输入法的任务就是从敲入的一系列字符尽可能的猜测要写的一段话,并把最可能的词语放在最前面让选择,这就可以看做一个 HMM 模型了。

再举一个例子,假如老师上课讲课,老师发出的一串连续的声音就是观测序列,而老师实际要表达的一段话就是隐藏状态序列。学生大脑的任务,就是从这一串连续的声音中判断出我最可能要表达的话的内容。

从这些例子中,我们可以发现,HMM 模型可以无处不在。但是上面的描述还不精确,下面我们用精确的数学符号来表述 HMM 模型。

3.2 HMM 模型的定义

对于 HMM 模型,首先我们假设 Q Q Q 是所有可能的隐藏状态的集合, V V V 是所有可能的观测状态的集合,即:

Q = q 1 , q 2 , . . . , q N V = v 1 , v 2 , . . . , v M \begin{aligned} & Q = q_1, q_2, ..., q_N\\ & V = v_1, v_2, ..., v_M \end{aligned} Q=q1,q2,...,qNV=v1,v2,...,vM

其中:

  • N N N 是可能的隐藏状态数
  • M M M 是所有的可能的观察状态数

对于一个长度为 T T T 的序列, i i i 是对应的隐藏状态序列, O O O 是对应的观察序列,即:

i = i 1 , i 2 , . . . , i T ( 几个面的骰子 ) O = o 1 , o 2 , . . . , o T ( 投掷骰子的结果 ) \begin{aligned} & i = i_1, i_2, ..., i_T \quad (几个面的骰子)\\ & O = o_1, o_2, ..., o_T \quad (投掷骰子的结果) \end{aligned} i=i1,i2,...,iT(几个面的骰子)O=o1,o2,...,oT(投掷骰子的结果)

其中:

  • 任意一个隐藏状态 i t ∈ Q i_t \in Q itQ
  • 任意一个观察状态 o t ∈ V o_t \in V otV

HMM 模型做了两个很重要的假设:

  1. 齐次马尔科夫链假设
  2. 观测独立性假设

3.2.1 【假设1】齐次马尔科夫链假设

任意时刻的隐藏状态只依赖于它前一个隐藏状态(时刻 t t t 只与 时刻 t − 1 t - 1 t1 有关)。

当然这样假设有点极端,因为很多时候我们的某一个隐藏状态不仅仅只依赖于前一个隐藏状态,可能还会依赖前两个或者是前三个。

但是这样假设的好处就是模型简单,便于求解。

如果在时刻 t t t 的隐藏状态是 i t = q i i_t = q_i it=qi,在时刻 t + 1 t + 1 t+1 的隐藏状态是 i t + 1 = q j i_{t + 1} = q_j it+1=qj,则从时刻 t t t 到时刻 t + 1 t+1 t+1 的 HMM 状态转移概率 a i j a_{ij} aij 可以表示为:

a i j = P ( i t + 1 = q j ∣ i t = q i ) a_{ij} = P(i_{t+1} = q_j | i_t = q_i) aij=P(it+1=qjit=qi)

这样 a i j a_{ij} aij 可以组成马尔科夫链的状态转移矩阵 A A A

A = [ a i j ] N × N A = [a_{ij}]_{N \times N} A=[aij]N×N

3.2.2 【假设2】观测独立性假设

任意时刻的观察状态只仅仅依赖于当前时刻的隐藏状态(跟其他时刻无关),这也是一个为了简化模型的假设(时刻 t t t 只与 时刻 t t t 有关)。

如果在时刻 t t t 的隐藏状态是 i t = q j i_t = q_j it=qj,而对应的观察状态为 o t = v k o_t = v_k ot=vk,则该时刻观察状态 v k v_k vk 在隐藏状态 q j q_j qj 下生成的概率为 b j ( k ) b_j(k) bj(k) 满足:

b j ( k ) = P ( o t = v k ∣ i t = q j ) b_j(k) = P(o_t = v_k | i_t = q_j) bj(k)=P(ot=vkit=qj)

这样 b j ( k ) b_j(k) bj(k) 可以组成观测状态生成的概率矩阵 B B B

B = [ b j ( k ) ] N × M B = [b_j(k)]_{N \times M} B=[bj(k)]N×M

除此之外,我们需要一组在时刻 t = 1 t=1 t=1 的隐藏状态概率分布 Π \Pi Π

Π = [ Π i ] N \Pi = [\Pi_{i}^{}]_{N} Π=[Πi]N

其中: Π i = P ( i 1 = q i ) \Pi_{i} = P(i_1 = q_i) Πi=P(i1=qi)


因此我们可以知道,一个 HMM 模型,可以由隐藏状态初始概率分布 Π \Pi Π,状态转移概率矩阵 A A A 和观测状态概率矩阵 B B B 三部分决定

  • 初始状态概率分布 Π \Pi Π 和 状态序列 A A A 决定状态序列
  • 观测序列 B B B 决定观测序列

因此,HMM 模型可以由一个三元组 λ \lambda λ 表示:

λ = ( A , B , Π ) = ( 状态序列 , 观测序列 , 初始状态概率分布 ) \lambda = (A, B, \Pi) = (状态序列, 观测序列, 初始状态概率分布) λ=(A,B,Π)=(状态序列,观测序列,初始状态概率分布)

3.3 一个 HMM 模型实例

下面我们用一个简单的实例来描述上面抽象出的 HMM 模型。这是一个盒子与球的模型。

例子来源于李航的《统计学习方法》

假设我们有 3 个盒子,每个盒子里都有 红色白色 两种球,这三个盒子里球的数量分别是:

盒子123
红球数547
白球数563

按照下面的方法从盒子里抽球,开始的时候:

  • 从第 1 个盒子抽球的概率是 0.2
  • 从第 2 个盒子抽球的概率是 0.4
  • 从第 3 个盒子抽球的概率是 0.4

以这个概率抽一次球后,将球放回。

然后从当前盒子转移到下一个盒子进行抽球。规则是:

  • 如果当前抽球的盒子是第 1 个盒子.则以 0.5 的概率仍然留在第 1 个盒子继续抽球,以 0.2 的概率去第 2 个盒子抽球,以 0.3 的概率去第 3 个盒子抽球。
  • 如果当前抽球的盒子是第 2 个盒子,则以 0.5 的概率仍然留在第 2 个盒子继续抽球,以 0.3 的概率去第 1 个盒子抽球,以 0.2 的概率去第 3 个盒子抽球。
  • 如果当前抽球的盒子是第 3 个盒子.则以 0.5 的概率仍然留在第 3 个盒子继续抽球,以 0.2 的概率去第 1 个盒子抽球,以 0.3 的概率去第 2 个盒子抽球。

如此下去,直到重复 3 次,得到一个球的颜色的观测序列 O O O

O = { 红 , 白 , 红 } O = \{ 红, 白, 红 \} O={,,}

注意在这个过程中,观察者只能看到球的颜色序列,却不能看到球是从哪个盒子里取出的

那么按照我们前面 HMM 模型的定义,我们的观察状态集合 V V V 是:

V = { 红 , 白 } M = 2 \begin{aligned} & V = \{ 红, 白 \}\\ & M = 2 \end{aligned} V={,}M=2

我们的隐藏状态集合 Q Q Q 是:

Q = { 盒子 1 , 盒子 2 , 盒子 3 } N = 3 \begin{aligned} & Q = \{ 盒子1, 盒子2, 盒子3 \}\\ & N = 3 \end{aligned} Q={盒子1,盒子2,盒子3}N=3

而观察序列 O O O(三个盒子)和状态序列 i i i(红白红)的长度 T T T 为都为 3。

初始状态概率分布 Π \Pi Π 为:

Π = ( 0.2 , 0.4 , 0.4 ) T \Pi = (0.2, 0.4, 0.4)^T Π=(0.2,0.4,0.4)T

从第 1 个盒子抽球的概率是 0.2;从第 2 个盒子抽球的概率是 0.4;从第 3 个盒子抽球的概率是 0.4

状态转移概率分布矩阵 A A A(不可见的,隐含的)为:

A = [ 0.5 0.2 0.3 0.3 0.5 0.2 0.2 0.3 0.5 ] N × N = 3 × 3 A = \begin{bmatrix} 0.5 & 0.2 & 0.3\\ 0.3 & 0.5 & 0.2\\ 0.2 & 0.3 & 0.5 \end{bmatrix}_{N \times N = 3 \times 3} A= 0.50.30.20.20.50.30.30.20.5 N×N=3×3

行表示第几次抽球(从2开始);列表示使用第几个盒子的概率

观测状态概率矩阵 B B B(可见的)为:

B = [ 0.5 0.5 0.4 0.6 0.7 0.3 ] N × M = 3 × 2 B = \begin{bmatrix} 0.5 & 0.5\\ 0.4 & 0.6\\ 0.7 & 0.3 \end{bmatrix}_{N \times M = 3 \times 2} B= 0.50.40.70.50.60.3 N×M=3×2

行代表第几个盒子;列1代表红球的概率,列2代表白球的概率

其中:

  • M M M 是所有的可能的观察状态数
  • N N N 是可能的隐藏状态数
  • V V V 是所有可能的观测状态的集合
  • Q Q Q 是所有可能的隐藏状态的集合
  • i i i 是状态序列
  • O O O 是观测序列
  • T T T 是序列的长度
  • A A A 是状态转移概率分布矩阵
  • B B B 是观测转移概率分布矩阵

3.4 HMM 观测序列 O O O 的生成

从上面的例子,我们也可以抽象出 HMM 观测序列 O O O 生成的过程。

  • 输入
    • HMM 的模型 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π)
    • 观测序列 O O O 的长度为 T T T
  • 输出
    • 观测序列 O = o 1 , o 2 , . . . , o T O = o_1, o_2,..., o_T O=o1,o2,...,oT

生成的过程如下

  1. 根据初始状态概率分布 Π \Pi Π 生成隐藏状态 i 1 i_1 i1
  2. 对于 t t t 1 1 1 T T T
    • a. 按照隐藏状态 i t i_t it 的观测状态分布 b i t ( k ) b_{it}(k) bit(k) 生成观察状态 o t o_t ot
    • b. 按照隐藏状态 i t i_t it 的状态转移概率分布 a i t a_{it} ait i t + 1 i_{t+1} it+1 产生隐藏状态 i t + 1 i_{t+1} it+1

所有的 o t o_t ot 一起形成观测序列 O = o 1 , o 2 , . . . , o T O = o_1, o_2,..., o_T O=o1,o2,...,oT

在隐马尔科夫模型(HMM)中, λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π) 表示 HMM 模型,其中 A A A 是状态转移概率矩阵, B B B 是观测概率矩阵, Π \Pi Π 是初始状态概率分布。 T T T 表示观测序列 O O O 的长度。 O = o 1 , o 2 , . . . , o T O = o_1, o_2,..., o_T O=o1,o2,...,oT 表示观测序列,其中 o t o_t ot 表示在时间步长 t t t 时的观测状态。 i t i_t it 表示在时间步长 t t t 时的隐藏状态。 b i t ( k ) b_{it}(k) bit(k) 表示在隐藏状态 i t i_t it 时,观测状态为 k k k 的概率。 a i t a_{it} ait 表示在隐藏状态 i t i_t it 时,转移到下一个隐藏状态的概率分布。这些字母代表了隐马尔科夫模型中的各种参数和变量。

3.5 HMM 模型的三个基本问题

HMM 模型一共有三个经典的问题需要解决:

  1. 评估观察序列概率:前向后向的概率计算
  2. 预测问题(也称为解码问题):维特比(Viterbi)算法
  3. 模型参数学习问题:鲍姆-韦尔奇(Baum-Welch)算法(状态未知)

3.5.1 【问题1】评估观察序列概率:前向后向的概率计算

即给定模型 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π) 和观测序列 O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT},计算在模型 λ \lambda λ 下某一个观测序列 O O O 出现的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

这个问题的求解需要用到前向后向算法,是 HMM 模型三个问题中最简单的。

3.5.2 【问题2】预测问题(也称为解码问题):维特比(Viterbi)算法

即给定模型 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π) 和观测序列 O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT},求给定观测序列条件 O O O 下,最可能出现的对应的状态序列 i i i

这个问题的求解需要用到基于动态规划的维特比算法,是 HMM 模型三个问题中复杂度居中的算法。

3.5.3 【问题3】模型参数学习问题:鲍姆-韦尔奇(Baum-Welch)算法(状态未知)

即给定观测序列 O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT},估计模型 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π) 的参数,使该模型下观测序列 O O O 的条件概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ) 最大。

这个问题的求解需要用到基于 EM 算法的鲍姆-韦尔奇算法,是 HMM 模型三个问题中最复杂的。

接下来的三节,我们将基于这个三个问题展开讨论。


小结

  • 什么样的问题可以用 HMM 模型解决?
    • 基于序列的,比如时间序列
    • 问题中包含两类数据:
      • 一类是可以观测到的观测序列
      • 另一类是不能观察到的隐藏状态序列
  • HMM 模型的两个重要假设:
    • 齐次马尔科夫链假设
    • 观测独立性假设
  • HMM 模型的三个基本问题:
    • 【问题1】评估观察序列概率:前向后向的概率计算
    • 【问题2】预测问题(也称为解码问题):维特比(Viterbi)算法
    • 【问题3】模型参数学习问题:鲍姆-韦尔奇(Baum-Welch)算法(状态未知)

4. 前向后向算法评估观察序列概率

学习目标

  • 知道用前向算法求 HMM 观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)
  • 知道用后向算法求 HMM 观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

本节我们就关注 HMM 第一个基本问题的解决方法,即已知模型 λ \lambda λ 和观测序列 O O O,求观测序列出现的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

4.1 回顾 HMM 问题一:求观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

首先我们回顾下 HMM 模型的问题二。这个问题是这样的:

我们已知 HMM 模型的参数 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π)。其中 A A A 是隐藏状态转移概率的矩阵, B B B 是观测状态生成概率的矩阵, Π \Pi Π 是隐藏状态的初始概率分布。同时我们也已经得到了观测序列 O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT},现在我们要求观测序列 O O O 在模型 λ \lambda λ 下出现的条件概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

乍一看,这个问题很简单。因为我们知道所有的隐藏状态之间的转移概率 A A A 和所有从隐藏状态到观测状态生成概率 B B B,那么我们是可以暴力求解的。

我们可以列举出所有可能出现的长度为 T T T 的隐藏序列 i = { i 1 , i 2 , . . . , i T } i=\{i_1,i_2, ..., i_T\} i={i1,i2,...,iT},分别求出这些隐藏序列 i i i 与观测序列 O = { o 1 , o 2 , . . . , o T } O =\{o_1,o_2,...,o_T\} O={o1,o2,...,oT} 的联合概率分布 P ( O , i ∣ λ ) P(O,i|\lambda) P(O,iλ),这样我们就可以很容易的求出边缘分布 P ( O ∣ λ ) P(O|\lambda) P(Oλ) 了。


具体暴力求解的方法是这样的:

  • 首先,任意隐藏序列 i = { i 1 , i 2 , . . . , i T } i=\{i_1,i_2, ..., i_T\} i={i1,i2,...,iT} 出现的概率是:出现的概率是: P ( i ∣ λ ) = Π i 1 a i 1 , i 2 a i 2 , i 3 . . . a i T − 1 , i T P(i|\lambda) = \Pi_{i_1} a_{i_1,i_2} a_{i_2,i_3} ... a_{i_{T-1}, i_T} P(iλ)=Πi1ai1,i2ai2,i3...aiT1,iT其中 Π \Pi Π是初始状态概率, a i t − 1 i t a_{i_{t-1}i_t} ait1it 是隐藏状态转移概率。

  • 对于固定的状态序列 i = { i 1 , i 2 , . . . , i T } i=\{i_1,i_2, ..., i_T\} i={i1,i2,...,iT},我们要求的观察序列 O = { o 1 , o 2 , . . . , o T } O =\{o_1,o_2,...,o_T\} O={o1,o2,...,oT} 出现的概率是: P ( O ∣ i , λ ) = b i 1 ( o 1 ) × b i 2 ( o 2 ) × . . . × b i T ( o T ) P(O|i, \lambda) = b_{i_1}(o_1)\times b_{i_2}(o_2) \times ... \times b_{i_T}(o_T) P(Oi,λ)=bi1(o1)×bi2(o2)×...×biT(oT)其中, b i t ( o t ) b_{i_t}(o_t) bit(ot) 是在隐藏状态 i t i_t it下观察到 o t o_t ot的概率

  • O O O i i i 联合出现的概率是: P ( O , i ∣ λ ) = P ( i ∣ λ ) P ( O ∣ i , λ ) = Π i 1 × b i 1 ( o 1 ) × a i 1 i 2 b i 2 ( o 2 ) . . . a i T − 1 i T b i T ( o T ) P(O, i|\lambda) = P(i|\lambda)P(O|i,\lambda) = \Pi_{i_1} \times b_{i_1}(o_1) \times a_{i_1i_2}b_{i_2}(o_2)...a_{i_{T-1}i_T}b_{i_T}(o_T) P(O,iλ)=P(iλ)P(Oi,λ)=Πi1×bi1(o1)×ai1i2bi2(o2)...aiT1iTbiT(oT)其中, Π \Pi Π是初始状态概率, a i t − 1 i t a_{i_{t-1}i_t} ait1it 是隐藏状态转移概率, b i t ( o t ) b_{i_t}(o_t) bit(ot) 是在隐藏状态 i t i_t it下观察到 o t o_t ot的概率

  • 然后求边缘概率分布,即可得到观测序列 O O O 在模型 λ \lambda λ 下出现的条件概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ) P ( O ∣ λ ) = ∑ i P ( O , i ∣ λ ) = ∑ i 1 , i 2 , . . . , i T Π i 1 × b i 1 ( o 1 ) × a i 1 i 2 × b i 2 ( o 2 ) × . . . × a i T − 1 i T × b i T ( O T ) P(O|\lambda) = \sum_i P(O, i|\lambda) = \sum_{i_1, i_2, ..., i_T}\Pi_{i_1}\times b_{i_1}(o_1)\times a_{i_1i_2}\times b_{i_2}(o_2) \times ... \times a_{i_{T-1}i_T} \times b_{i_T}(O_T) P(Oλ)=iP(O,iλ)=i1,i2,...,iTΠi1×bi1(o1)×ai1i2×bi2(o2)×...×aiT1iT×biT(OT)其中, Π \Pi Π是初始状态概率, a i t − 1 i t a_{i_{t-1}i_t} ait1it 是隐藏状态转移概率, b i t ( o t ) b_{i_t}(o_t) bit(ot) 是在隐藏状态 i t i_t it下观察到 o t o_t ot的概率

虽然上述方法有效,但是如果我们的隐藏状态数 N N N 非常多的那就麻烦了,此时我们预测状态有 N T N^T NT 种组合,算法的时间复杂度是 O ( T N T ) O(TN^T) O(TNT) 阶的。

因此:

  • 对于一些隐藏状态数 N N N 极少的模型,我们可以用暴力求解法来得到观测序列出现的概率
  • 但如果隐藏状态数 N N N 多,上述算法太耗时,我们需要寻找其他简洁的算法。

前向后向算法就是来帮助我们在较低的时间复杂度情况下求解这个问题的。

4.2 用前向算法求 HMM 观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

前向后向算法 是 前向算法 和 后向算法 的统称,这两个算法都可以用来求 HMM 观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)。我们先来看看 前向算法 是如何求解这个问题的。

4.2.1 流程梳理

前向算法 本质上属于 动态规划的算法,也就是我们要通过找到 局部状态 递推的公式,这样一步步的从 子问题的最优解 拓展到 整个问题的最优解。

动态规划算法的思想:在算的过程中保证前面的步骤是最优的(即当前结果最优

在前向算法中,通过定义“前向概率”来定义动态规划的这个局部状态。

那么什么是 前向概率 呢?其实定义很简单:定义时刻 t t t 时隐藏状态为 q i q_i qi,那么观测状态的序列为 o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot 的概率就是 前向概率。记为:

α t ( i ) = P ( o 1 , o 2 , . . . , o t , i t = q i ∣ λ ) \alpha_t(i) = P(o_1, o_2, ..., o_t, i_t = q_i | \lambda) αt(i)=P(o1,o2,...,ot,it=qiλ)

其中:

  • α t ( i ) \alpha_t(i) αt(i) 表示前向概率,即定义时刻 t t t 时隐藏状态为 q i q_i qi,那么观测状态的序列为 o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot 的概率。
  • o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot 表示观测状态的序列。
  • i t = q i i_t = q_i it=qi 表示时刻 t t t 时隐藏状态为 q i q_i qi
  • λ \lambda λ 表示隐马尔科夫模型的参数。

既然是动态规划,我们就要递推了,现在假设我们已经找到了在时刻 t t t 时各个隐藏状态的前向概率,现在我们需要递推出时刻 t + 1 t+1 t+1 时各个隐藏状态的前向概率。

我们可以基于时刻 t t t 时各个隐藏状态的前向概率 α t \alpha_t αt,再乘以对应的状态转移概率 a j i a_{ji} aji,即 α t ( j ) × a j i \alpha_t(j)\times a_{ji} αt(j)×aji 就是在时刻 t t t 观测到 o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot,并且时刻 t t t 隐藏状态 q j q_j qj,时刻 t + 1 t+1 t+1 隐藏状态 q i q_i qi 的概率。

Q:为什么是 a j i a_{ji} aji 而不是 a t a_t at
A a j i a_{ji} aji 表示的是从隐藏状态 q j q_j qj 转移到隐藏状态 q i q_i qi 的概率,它是隐马尔科夫模型的参数之一。在隐马尔科夫模型中,状态转移概率矩阵 A = [ a i j ] A = [a_{ij}] A=[aij] 是固定不变的,不随时间变化。所以,在递推计算前向概率时,我们使用固定的状态转移概率矩阵 A A A 中的元素 a j i a_{ji} aji,而不是随时间变化的 a t a_t at

如果将下面所有的线对应的概率求和,即 ∑ j = 1 N α t ( j ) a j i \sum_{j=1}^N \alpha_t(j)a_{ji} j=1Nαt(j)aji 就是在时刻 t t t 观测到 o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot,并且时刻 t + 1 t+1 t+1 隐藏状态 q i q_i qi 的概率。

继续一步,由于观测状态 o t + 1 o_{t+1} ot+1 只依赖于 t + 1 t+1 t+1 时刻隐藏状态 q i q_i qi,这样 [ ∑ i = 1 N α t ( j ) a j i ] b i ( o t + 1 ) \left[\sum_{i=1}^N \alpha_t(j)a_{ji}\right]b_i(o_{t+1}) [i=1Nαt(j)aji]bi(ot+1) 就是在时刻 t + 1 t+1 t+1 观测到 o 1 , o 2 , . . . , o t + 1 o_1, o_2, ..., o_{t+1} o1,o2,...,ot+1,并且时刻 t + 1 t+1 t+1 隐藏状态 q i q_i qi 的概率。

而这个概率,恰恰就是时刻 t + 1 t+1 t+1 对应的隐藏状态 i i i 的前向概率,这样我们得到了前向概率的递推关系式,如下所示:

α t + 1 ( i ) = [ ∑ j = 1 N α t ( j ) a j i ] b i ( o t + 1 ) \alpha_{t+1}(i) = \left[ \sum_{j=1}^N \alpha_t(j)a_{ji} \right] b_i(o_{t+1}) αt+1(i)=[j=1Nαt(j)aji]bi(ot+1)

我们的动态规划从时刻 1 开始,到时刻 T T T 结束。由于 α T ( i ) \alpha T(i) αT(i) 表示在时刻 T T T 观测序列为 o 1 , o 2 , . . . , o T o_1, o_2, ..., o_T o1,o2,...,oT,并且时刻 T T T 隐藏状态 q i q_i qi 的概率,我们只要将所有隐藏状态对应的概率相加,即 i = ∑ i = 1 N α T ( i ) i=\sum_{i=1}^N \alpha_T(i) i=i=1NαT(i) 就得到了在时刻 T T T 观测序列为 o 1 , o 2 , . . . , o T o_1, o_2, ..., o_T o1,o2,...,oT 的概率。

4.2.2 算法总结

  • 输入:HMM 模型参数 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π),观测序列 O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT}
  • 输出:观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)
  1. 计算时刻 1 1 1 的各个隐藏状态前向概率: α 1 ( i ) = Π i b i ( o 1 ) i = 1 , 2 , . . . , N \alpha_1(i) = \Pi_i b_i(o_1)\quad i=1, 2, ..., N α1(i)=Πibi(o1)i=1,2,...,N
  2. 递推时刻 2 , 3 , . . . , T 2,3,..., T 2,3,...,T 的前向概率: α t + 1 ( i ) = [ ∑ j = 1 N α t ( j ) a j i ] b i ( o t + 1 ) i = 1 , 2 , . . . , N \alpha_{t+1}(i) = \left[ \sum_{j=1}^N \alpha_t(j)a_{ji} \right]b_i(o_{t+1}) \quad i = 1, 2, ..., N αt+1(i)=[j=1Nαt(j)aji]bi(ot+1)i=1,2,...,N
  3. 计算最终结果: P ( O ∣ λ ) = ∑ i = 1 N α T ( i ) P(O|\lambda) = \sum_{i=1}^N \alpha_T(i) P(Oλ)=i=1NαT(i)

其中:

  • λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π) 表示隐马尔科夫模型的参数
    • A A A 是状态转移概率矩阵
    • B B B 是观测概率矩阵
    • Π \Pi Π 是初始状态概率向量
  • O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT} 表示观测序列。
  • P ( O ∣ λ ) P(O|\lambda) P(Oλ) 表示在模型 λ \lambda λ 下观测序列 O O O 出现的概率。
  • α t ( i ) \alpha_t(i) αt(i) 表示前向概率,即定义时刻 t t t 时隐藏状态为 q i q_i qi,那么观测状态的序列为 o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot 的概率。
  • Π i \Pi_i Πi 表示初始状态概率向量中第 i i i 个元素的值。
  • b i ( o t ) b_i(o_t) bi(ot) 表示在隐藏状态 q i q_i qi 下观测到 o t o_t ot 的概率。
  • a j i a_{ji} aji 表示从隐藏状态 q j q_j qj 转移到隐藏状态 q i q_i qi 的概率。

从递推公式可以看出,我们的算法时间复杂度是 O ( T N 2 ) O(TN^2) O(TN2),比暴力解法的时间复杂度 O ( T N T ) O(TN^T) O(TNT) 少了几个数量级。

4.3 HMM 前向算法求解实例

这里我们用前面盒子与球的例子来显示前向概率 α \alpha α 的计算。我们的观察集合是:

V = { 红 , 白 } M = 2 \begin{aligned} & V = \{ 红,白 \}\\ & M = 2 \end{aligned} V={,}M=2

我们的状态集合是:

Q = { 盒子 1 , 盒子 2 , 盒子 3 } N = 3 \begin{aligned} & Q = \{盒子1, 盒子2, 盒子3\}\\ & N = 3 \end{aligned} Q={盒子1,盒子2,盒子3}N=3

而观察序列 O O O 和状态序列 i i i 的长度为都为 3。

初始状态分布为:

Π = ( 0.2 , 0.4 , 0.4 ) T \Pi = (0.2, 0.4, 0.4)^T Π=(0.2,0.4,0.4)T

状态转移概率分布矩阵 A A A(不可见的,隐含的)为:

A = [ 0.5 0.2 0.3 0.3 0.5 0.2 0.2 0.3 0.5 ] N × N = 3 × 3 A = \begin{bmatrix} 0.5 & 0.2 & 0.3\\ 0.3 & 0.5 & 0.2\\ 0.2 & 0.3 & 0.5 \end{bmatrix}_{N \times N = 3 \times 3} A= 0.50.30.20.20.50.30.30.20.5 N×N=3×3

行表示第几次抽球(从2开始);列表示使用第几个盒子的概率

观测状态概率矩阵 B B B(可见的)为:

B = [ 0.5 0.5 0.4 0.6 0.7 0.3 ] N × M = 3 × 2 B = \begin{bmatrix} 0.5 & 0.5\\ 0.4 & 0.6\\ 0.7 & 0.3 \end{bmatrix}_{N \times M = 3 \times 2} B= 0.50.40.70.50.60.3 N×M=3×2

行代表第几个盒子;列1代表红球的概率,列2代表白球的概率

球的颜色的观测序列:

O = { 红 , 白 , 红 } O = \{红, 白, 红\} O={,,}


按照我们上一节的前向算法。首先计算时刻 1 三个状态的前向概率 α 1 ( i ) \alpha_1(i) α1(i)

时刻 1 是红色球

  • 隐藏状态是盒子 1 的概率为: α 1 ( 1 ) = Π 1 b 1 ( o 1 ) = 0.2 抽到盒子 1 的概率 × 0.5 抽到红球的概率 = 0.1 \alpha_1(1) = \Pi_1b_1(o_1) = \underset{抽到盒子1的概率}{0.2} \times \underset{抽到红球的概率}{0.5} = 0.1 α1(1)=Π1b1(o1)=抽到盒子1的概率0.2×抽到红球的概率0.5=0.1
  • 隐藏状态是盒子 2 的概率为: α 1 ( 2 ) = Π 2 b 2 ( o 1 ) = 0.4 抽到盒子 2 的概率 × 0.4 抽到红球的概率 = 0.16 \alpha_1(2) = \Pi_2b_2(o_1) = \underset{抽到盒子2的概率}{0.4} \times \underset{抽到红球的概率}{0.4} = 0.16 α1(2)=Π2b2(o1)=抽到盒子2的概率0.4×抽到红球的概率0.4=0.16
  • 隐藏状态是盒子 3 的概率为: α 1 ( 3 ) = Π 3 b 3 ( o 1 ) = 0.4 抽到盒子 3 的概率 × 0.7 抽到红球的概率 = 0.28 \alpha_1(3) = \Pi_3b_3(o_1) = \underset{抽到盒子3的概率}{0.4} \times \underset{抽到红球的概率}{0.7} = 0.28 α1(3)=Π3b3(o1)=抽到盒子3的概率0.4×抽到红球的概率0.7=0.28

现在我们可以开始递推了,首先递推时刻 2 三个状态的前向概率 α 2 ( i ) \alpha_2(i) α2(i)

时刻 2 是白色球

  • 隐藏状态是盒子 1 的概率为: α 2 ( 1 ) = [ ∑ i = 1 3 α 1 ( i ) a i 1 ] b 1 ( o 2 ) = [ 0.1 第一次是盒子 1 × 0.5 盒子 1 → 盒子 1 ‾ 第一种情况 + 0.16 第一次是盒子 2 × 0.3 盒子 2 → 盒子 1 ‾ 第二种情况 + 0.28 第一次是盒子 3 × 0.2 盒子 3 → 盒子 1 ‾ 第三种情况 ] × 0.5 抽到白球 = 0.077 \begin{aligned}\alpha_2(1) & = \left[ \sum_{i=1}^3 \alpha_1(i) a_{i1} \right]b_1(o_2)\\ & = \left[\underset{第一种情况}{\underline{\underset{第一次是盒子1}{0.1} \times \underset{盒子1\rightarrow盒子1}{0.5}}} + \underset{第二种情况}{\underline{\underset{第一次是盒子2}{0.16} \times \underset{盒子2\rightarrow盒子1}{0.3}}} + \underset{第三种情况}{\underline{\underset{第一次是盒子3}{0.28} \times \underset{盒子3\rightarrow盒子1}{0.2}}} \right] \times \underset{抽到白球}{0.5}\\& = 0.077\end{aligned} α2(1)=[i=13α1(i)ai1]b1(o2)= 第一种情况第一次是盒子10.1×盒子1盒子10.5+第二种情况第一次是盒子20.16×盒子2盒子10.3+第三种情况第一次是盒子30.28×盒子3盒子10.2 ×抽到白球0.5=0.077
  • 隐藏状态是盒子 2 的概率为: α 2 ( 2 ) = [ ∑ i = 1 3 α 1 ( i ) a i 2 ] b 2 ( o 2 ) = [ 0.1 第一次是盒子 1 × 0.2 盒子 1 → 盒子 2 ‾ 第一种情况 + 0.16 第一次是盒子 2 × 0.5 盒子 2 → 盒子 2 ‾ 第二种情况 + 0.28 第一次是盒子 3 × 0.3 盒子 3 → 盒子 2 ‾ 第三种情况 ] × 0.6 抽到白球 = 0.1104 \begin{aligned}\alpha_2(2) & = \left[ \sum_{i=1}^3 \alpha_1(i) a_{i2} \right]b_2(o_2)\\ & = \left[\underset{第一种情况}{\underline{\underset{第一次是盒子1}{0.1} \times \underset{盒子1\rightarrow盒子2}{0.2}}} + \underset{第二种情况}{\underline{\underset{第一次是盒子2}{0.16} \times \underset{盒子2\rightarrow盒子2}{0.5}}} + \underset{第三种情况}{\underline{\underset{第一次是盒子3}{0.28} \times \underset{盒子3\rightarrow盒子2}{0.3}}} \right] \times \underset{抽到白球}{0.6}\\& = 0.1104\end{aligned} α2(2)=[i=13α1(i)ai2]b2(o2)= 第一种情况第一次是盒子10.1×盒子1盒子20.2+第二种情况第一次是盒子20.16×盒子2盒子20.5+第三种情况第一次是盒子30.28×盒子3盒子20.3 ×抽到白球0.6=0.1104
  • 隐藏状态是盒子 3 的概率为: α 2 ( 3 ) = [ ∑ i = 1 3 α 1 ( i ) a i 3 ] b 3 ( o 2 ) = [ 0.1 第一次是盒子 1 × 0.3 盒子 1 → 盒子 3 ‾ 第一种情况 + 0.16 第一次是盒子 2 × 0.2 盒子 2 → 盒子 3 ‾ 第二种情况 + 0.28 第一次是盒子 3 × 0.5 盒子 3 → 盒子 3 ‾ 第三种情况 ] × 0.3 抽到白球 = 0.0606 \begin{aligned}\alpha_2(3) & = \left[ \sum_{i=1}^3 \alpha_1(i) a_{i3} \right]b_3(o_2)\\ & = \left[\underset{第一种情况}{\underline{\underset{第一次是盒子1}{0.1} \times \underset{盒子1\rightarrow盒子3}{0.3}}} + \underset{第二种情况}{\underline{\underset{第一次是盒子2}{0.16} \times \underset{盒子2\rightarrow盒子3}{0.2}}} + \underset{第三种情况}{\underline{\underset{第一次是盒子3}{0.28} \times \underset{盒子3\rightarrow盒子3}{0.5}}} \right] \times \underset{抽到白球}{0.3}\\& = 0.0606\end{aligned} α2(3)=[i=13α1(i)ai3]b3(o2)= 第一种情况第一次是盒子10.1×盒子1盒子30.3+第二种情况第一次是盒子20.16×盒子2盒子30.2+第三种情况第一次是盒子30.28×盒子3盒子30.5 ×抽到白球0.3=0.0606

在计算时刻 2 时只考虑时刻 1


继续递推,现在我们递推时刻 3 三个状态的前向概率 α 2 ( i ) \alpha_2(i) α2(i)

时刻 3 是红色球

  • 隐藏状态是盒子 1 的概率为: α 3 ( 1 ) = [ ∑ i = 1 3 α 2 ( i ) a i 1 ] b 1 ( o 3 ) = [ 0.077 第一次是盒子 1 + 1 × 0.5 盒子 1 → 盒子 1 ‾ 第一种情况 + 0.1104 第一次是盒子 1 + 2 × 0.3 盒子 2 → 盒子 1 ‾ 第二种情况 + 0.0606 第一次是盒子 1 + 3 × 0.2 盒子 3 → 盒子 1 ‾ 第三种情况 ] × 0.3 抽到红球 = 0.04187 \begin{aligned} \alpha_3(1) & = \left[ \sum_{i=1}^3 \alpha_2(i)a_{i1} \right] b_1(o_3) \\ & = \left[\underset{第一种情况}{\underline{\underset{第一次是盒子1+1}{0.077} \times \underset{盒子1\rightarrow盒子1}{0.5}}} + \underset{第二种情况}{\underline{\underset{第一次是盒子1+2}{0.1104} \times \underset{盒子2\rightarrow盒子1}{0.3}}} + \underset{第三种情况}{\underline{\underset{第一次是盒子1+3}{0.0606} \times \underset{盒子3\rightarrow盒子1}{0.2}}} \right] \times \underset{抽到红球}{0.3}\\& = 0.04187 \end{aligned} α3(1)=[i=13α2(i)ai1]b1(o3)= 第一种情况第一次是盒子1+10.077×盒子1盒子10.5+第二种情况第一次是盒子1+20.1104×盒子2盒子10.3+第三种情况第一次是盒子1+30.0606×盒子3盒子10.2 ×抽到红球0.3=0.04187
  • 隐藏状态是盒子 2 的概率为: α 3 ( 2 ) = [ ∑ i = 1 3 α 2 ( i ) a i 2 ] b 2 ( o 3 ) = [ 0.077 第一次是盒子 1 + 1 × 0.2 盒子 1 → 盒子 2 ‾ 第一种情况 + 0.1104 第一次是盒子 1 + 2 × 0.5 盒子 2 → 盒子 2 ‾ 第二种情况 + 0.0606 第一次是盒子 1 + 3 × 0.3 盒子 3 → 盒子 2 ‾ 第三种情况 ] × 0.4 抽到红球 = 0.03551 \begin{aligned} \alpha_3(2) & = \left[ \sum_{i=1}^3 \alpha_2(i)a_{i2} \right] b_2(o_3) \\ & = \left[\underset{第一种情况}{\underline{\underset{第一次是盒子1+1}{0.077} \times \underset{盒子1\rightarrow盒子2}{0.2}}} + \underset{第二种情况}{\underline{\underset{第一次是盒子1+2}{0.1104} \times \underset{盒子2\rightarrow盒子2}{0.5}}} + \underset{第三种情况}{\underline{\underset{第一次是盒子1+3}{0.0606} \times \underset{盒子3\rightarrow盒子2}{0.3}}} \right] \times \underset{抽到红球}{0.4}\\& = 0.03551 \end{aligned} α3(2)=[i=13α2(i)ai2]b2(o3)= 第一种情况第一次是盒子1+10.077×盒子1盒子20.2+第二种情况第一次是盒子1+20.1104×盒子2盒子20.5+第三种情况第一次是盒子1+30.0606×盒子3盒子20.3 ×抽到红球0.4=0.03551
  • 隐藏状态是盒子 3 的概率为: α 3 ( 3 ) = [ ∑ i = 1 3 α 3 ( i ) a i 3 ] b 3 ( o 3 ) = [ 0.077 第一次是盒子 1 + 1 × 0.3 盒子 1 → 盒子 3 ‾ 第一种情况 + 0.1104 第一次是盒子 1 + 2 × 0.2 盒子 2 → 盒子 3 ‾ 第二种情况 + 0.0606 第一次是盒子 1 + 3 × 0.5 盒子 3 → 盒子 3 ‾ 第三种情况 ] × 0.3 抽到红球 = 0.05284 \begin{aligned} \alpha_3(3) & = \left[ \sum_{i=1}^3 \alpha_3(i)a_{i3} \right] b_3(o_3) \\ & = \left[\underset{第一种情况}{\underline{\underset{第一次是盒子1+1}{0.077} \times \underset{盒子1\rightarrow盒子3}{0.3}}} + \underset{第二种情况}{\underline{\underset{第一次是盒子1+2}{0.1104} \times \underset{盒子2\rightarrow盒子3}{0.2}}} + \underset{第三种情况}{\underline{\underset{第一次是盒子1+3}{0.0606} \times \underset{盒子3\rightarrow盒子3}{0.5}}} \right] \times \underset{抽到红球}{0.3}\\& = 0.05284 \end{aligned} α3(3)=[i=13α3(i)ai3]b3(o3)= 第一种情况第一次是盒子1+10.077×盒子1盒子30.3+第二种情况第一次是盒子1+20.1104×盒子2盒子30.2+第三种情况第一次是盒子1+30.0606×盒子3盒子30.5 ×抽到红球0.3=0.05284

在计算时刻 3 时只考虑时刻 2


最终我们求出观测序列: O = { 红 , 白 , 红 } O = \{红, 白, 红\} O={,,} 的概率为:

P ( O ∣ λ ) = ∑ i = 1 3 α 3 ( i ) = 0.04187 + 0.03551 + 0.05284 = 0.13022 \begin{aligned} P(O|\lambda) & = \sum_{i=1}^3 \alpha_3(i) \\ & = 0.04187 + 0.03551 + 0.05284 \\ & = 0.13022 \end{aligned} P(Oλ)=i=13α3(i)=0.04187+0.03551+0.05284=0.13022

在计算的时候,我们只考虑时刻的前一步,这就是前向算法。

需要注意的是,我们一定要保证当前时刻一定是局部最优解(动态规划的思想)

4.4 用后向算法求 HMM 观测序列的概率

4.4.1 流程梳理

熟悉了用前向算法求 HMM 观测序列的概率,现在我们再来看看怎么用后向算法求 HMM 观测序列的概率。

后向算法和前向算法非常类似,都是用的动态规划,唯一的区别是选择的局部状态不同,后向算法用的是“后向概率”。

简单来说,前向是从 1 1 1 T T T 的算法,而后向算法是 从 T T T 1 1 1 的算法。

4.4.2 后向算法流程

以下是后向算法的流程,注意和前向算法的相同点及不同点:

  • 输入:HMM 模型 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π),观测序列 O = ( o 1 , o 2 , . . . , o T ) O=(o_1, o_2,..., o_T) O=(o1,o2,...,oT)
  • 输出:观测序列概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

初始化时刻 T T T 的各个隐藏状态后向概率:

β T ( i ) = 1 i = 1 , 2 , . . . , N \beta_T(i) = 1 \quad i = 1, 2, ..., N βT(i)=1i=1,2,...,N

各个隐藏状态前向概率用 α \alpha α 表示;各个隐藏状态后向概率用 β \beta β 表示。

递推时刻 T − 1 , T − 2 , . . . , 1 T-1,T-2,..., 1 T1,T2,...,1 时刻的后向概率:

β t ( i ) = ∑ j = 1 N a i j b j ( o t + 1 ) β t + 1 ( j ) i = 1 , 2 , . . . , N \beta_t(i) = \sum_{j = 1}^N a_{ij}b_j(o_{t+1})\beta_{t+1}(j) \quad i = 1, 2, ..., N βt(i)=j=1Naijbj(ot+1)βt+1(j)i=1,2,...,N

计算最终结果:

P ( O ∣ λ ) = ∑ i = 1 N Π i b i ( o 1 ) β 1 ( i ) P(O | \lambda) = \sum_{i=1}^N \Pi_ib_i(o_1)\beta_1(i) P(Oλ)=i=1NΠibi(o1)β1(i)

其中:

  • A A A:状态转移矩阵,其中 a i j a_{ij} aij 表示从隐藏状态 i i i 转移到隐藏状态 j j j 的概率。
  • B B B:观测概率矩阵,其中 b j ( k ) b_j(k) bj(k) 表示在隐藏状态 j j j 下观测到符号 k k k 的概率。
  • Π \Pi Π:初始状态概率向量,其中 π i \pi_i πi 表示初始时刻隐藏状态为 i i i 的概率。
  • O O O:观测序列,其中 o t o_t ot 表示时刻 t t t 的观测值。
  • λ \lambda λ:HMM 模型参数,包括状态转移矩阵 A A A、观测概率矩阵 B B B 和初始状态概率向量 Π \Pi Π
  • β t ( i ) \beta_t(i) βt(i):时刻 t t t 处于隐藏状态 i i i 且从时刻 t + 1 t+1 t+1 到时刻 T T T 的观测序列为 o t + 1 , o t + 2 , . . . , o T o_{t+1}, o_{t+2}, ..., o_T ot+1,ot+2,...,oT 的后向概率。

此时我们的算法时间复杂度仍然是 O ( T N 2 ) O(TN^2) O(TN2)

4.5 总结

4.5.1 前向算法求 HMM 观测序列 O O O

  • 输入:HMM 模型参数 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π),观测序列 O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT}
  • 输出:观测序列 O O O 的概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)
  1. 计算时刻 1 1 1 的各个隐藏状态前向概率: α 1 ( i ) = Π i b i ( o 1 ) i = 1 , 2 , . . . , N \alpha_1(i) = \Pi_i b_i(o_1)\quad i=1, 2, ..., N α1(i)=Πibi(o1)i=1,2,...,N
  2. 递推时刻 2 , 3 , . . . , T 2,3,..., T 2,3,...,T 的前向概率: α t + 1 ( i ) = [ ∑ j = 1 N α t ( j ) a j i ] b i ( o t + 1 ) i = 1 , 2 , . . . , N \alpha_{t+1}(i) = \left[ \sum_{j=1}^N \alpha_t(j)a_{ji} \right]b_i(o_{t+1}) \quad i = 1, 2, ..., N αt+1(i)=[j=1Nαt(j)aji]bi(ot+1)i=1,2,...,N
  3. 计算最终结果: P ( O ∣ λ ) = ∑ i = 1 N α T ( i ) P(O|\lambda) = \sum_{i=1}^N \alpha_T(i) P(Oλ)=i=1NαT(i)

其中:

  • λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π) 表示隐马尔科夫模型的参数
    • A A A 是状态转移概率矩阵
    • B B B 是观测概率矩阵
    • Π \Pi Π 是初始状态概率向量
  • O = { o 1 , o 2 , . . . , o T } O = \{o_1, o_2,..., o_T\} O={o1,o2,...,oT} 表示观测序列。
  • P ( O ∣ λ ) P(O|\lambda) P(Oλ) 表示在模型 λ \lambda λ 下观测序列 O O O 出现的概率。
  • α t ( i ) \alpha_t(i) αt(i) 表示前向概率,即定义时刻 t t t 时隐藏状态为 q i q_i qi,那么观测状态的序列为 o 1 , o 2 , . . . , o t o_1, o_2, ..., o_t o1,o2,...,ot 的概率。
  • Π i \Pi_i Πi 表示初始状态概率向量中第 i i i 个元素的值。
  • b i ( o t ) b_i(o_t) bi(ot) 表示在隐藏状态 q i q_i qi 下观测到 o t o_t ot 的概率。
  • a j i a_{ji} aji 表示从隐藏状态 q j q_j qj 转移到隐藏状态 q i q_i qi 的概率。

4.5.2 后向算法求 HMM 观测序列 O O O

  • 输入:HMM 模型 λ = ( A , B , Π ) \lambda = (A, B, \Pi) λ=(A,B,Π),观测序列 O = ( o 1 , o 2 , . . . , o T ) O=(o_1, o_2,..., o_T) O=(o1,o2,...,oT)
  • 输出:观测序列概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)

初始化时刻 T T T 的各个隐藏状态后向概率:

β T ( i ) = 1 i = 1 , 2 , . . . , N \beta_T(i) = 1 \quad i = 1, 2, ..., N βT(i)=1i=1,2,...,N

各个隐藏状态前向概率用 α \alpha α 表示;各个隐藏状态后向概率用 β \beta β 表示。

递推时刻 T − 1 , T − 2 , . . . , 1 T-1,T-2,..., 1 T1,T2,...,1 时刻的后向概率:

β t ( i ) = ∑ j = 1 N a i j b j ( o t + 1 ) β t + 1 ( j ) i = 1 , 2 , . . . , N \beta_t(i) = \sum_{j = 1}^N a_{ij}b_j(o_{t+1})\beta_{t+1}(j) \quad i = 1, 2, ..., N βt(i)=j=1Naijbj(ot+1)βt+1(j)i=1,2,...,N

计算最终结果:

P ( O ∣ λ ) = ∑ i = 1 N Π i b i ( o 1 ) β 1 ( i ) P(O | \lambda) = \sum_{i=1}^N \Pi_ib_i(o_1)\beta_1(i) P(Oλ)=i=1NΠibi(o1)β1(i)

其中:

  • A A A:状态转移矩阵,其中 a i j a_{ij} aij 表示从隐藏状态 i i i 转移到隐藏状态 j j j 的概率。
  • B B B:观测概率矩阵,其中 b j ( k ) b_j(k) bj(k) 表示在隐藏状态 j j j 下观测到符号 k k k 的概率。
  • Π \Pi Π:初始状态概率向量,其中 π i \pi_i πi 表示初始时刻隐藏状态为 i i i 的概率。
  • O O O:观测序列,其中 o t o_t ot 表示时刻 t t t 的观测值。
  • λ \lambda λ:HMM 模型参数,包括状态转移矩阵 A A A、观测概率矩阵 B B B 和初始状态概率向量 Π \Pi Π
  • β t ( i ) \beta_t(i) βt(i):时刻 t t t 处于隐藏状态 i i i 且从时刻 t + 1 t+1 t+1 到时刻 T T T 的观测序列为 o t + 1 , o t + 2 , . . . , o T o_{t+1}, o_{t+2}, ..., o_T ot+1,ot+2,...,oT 的后向概率。

4.5.3 对比

  • 前向和后向算法的输入和输出是一样的。
  • 前向算法先求时刻 1;而后向算法是先求时刻 T T T
  • 各个隐藏状态前向概率 α 1 ( i ) = Π i b i ( o i ) \alpha_1(i) = \Pi_ib_i(o_i) α1(i)=Πibi(oi)w;各个隐藏状态后向概率 β T ( i ) = 1 i = 1 , 2 , . . . , N \beta_T(i) = 1 \quad i = 1, 2, ..., N βT(i)=1i=1,2,...,N(直接认为是1)
  • 前向是从 1 到 T T T;而后向是从 T − 1 T-1 T1 到 1
  • 前向是一步步前进累加求解;而后向是一步步向后退求解
  • 最终结果求解一样
  • 时间复杂度二者一样,都是 O ( T N 2 ) O(TN^2) O(TN2)
  • 二者都使用了动态规划的思想(确保当前步是最优的(局部最优))

4.5.4 如何选择 前向算法 和 后向算法?

前向算法和后向算法都可以用来计算观测序列概率 P ( O ∣ λ ) P(O|\lambda) P(Oλ)。它们的时间复杂度都是 O ( T N 2 ) O(TN^2) O(TN2),因此在计算观测序列概率时,二者的效率相当。

不过,在某些情况下,前向算法和后向算法可以结合使用来解决其他问题。例如,在计算给定模型 λ \lambda λ 和观测序列 O O O 的条件下,时刻 t t t 处于隐藏状态 i i i 的概率时,可以使用前向概率 α t ( i ) \alpha_t(i) αt(i) 和后向概率 β t ( i ) \beta_t(i) βt(i) 来计算:

P ( i t = q i ∣ O , λ ) = α t ( i ) β t ( i ) ∑ j = 1 N α t ( j ) β t ( j ) P(i_t = q_i | O, \lambda) = \frac{\alpha_t(i)\beta_t(i)}{\sum_{j=1}^N\alpha_t(j)\beta_t(j)} P(it=qiO,λ)=j=1Nαt(j)βt(j)αt(i)βt(i)

因此,在选择前向算法还是后向算法时,应根据具体问题来决定

  • 如果只需要计算观测序列概率,那么二者都可以
  • 如果需要解决其他问题,可能需要结合使用前向算法和后向算法。

这篇关于[学习笔记] [机器学习] 12. [上] HMM 隐马尔可夫算法(马尔科夫链、HMM 三类问题、前后后向算法、维特比算法、鲍姆-韦尔奇算法、API 及实例)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

Java操作ElasticSearch的实例详解

《Java操作ElasticSearch的实例详解》Elasticsearch是一个分布式的搜索和分析引擎,广泛用于全文搜索、日志分析等场景,本文将介绍如何在Java应用中使用Elastics... 目录简介环境准备1. 安装 Elasticsearch2. 添加依赖连接 Elasticsearch1. 创

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

使用C#代码计算数学表达式实例

《使用C#代码计算数学表达式实例》这段文字主要讲述了如何使用C#语言来计算数学表达式,该程序通过使用Dictionary保存变量,定义了运算符优先级,并实现了EvaluateExpression方法来... 目录C#代码计算数学表达式该方法很长,因此我将分段描述下面的代码片段显示了下一步以下代码显示该方法如

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1