原创:hxj7
从本文开始,到了序列比对的第二部分,主要是介绍HMM以及其在序列比对中的基础应用。
前面八篇文章介绍了动态规划在序列比对中的基础应用。从本文开始,开始介绍HMM(隐马尔可夫模型)。
Markov链
首先我们先介绍马尔可夫链(Markov链),这个大家可能都熟悉,高中数学中就学过。简单来说,Markov链就是一系列状态构成,每一个状态出现的概率只与它前一个状态有关。
具体地说,假设有一系列状态,依次由构成,那么在已经发生的情况下,发生的概率只与相关。即:
由此,我们可以推导出:
为了建模的方便,一般还会假设一个初始状态,则上面的公式变成:
为了更形象说明,以投骰子为例来说,假设我们有下面这张图:
图片引自《生物序列分析》
这张图是说有两种骰子,一种是“公平的”(Fair,简写F),一种是“作弊的”(Loaded,简写L)。F骰子掷出1-6的概率是一样的,而L骰子掷出6的概率为0.5,其余1-5的概率都是0.1。
此外,如果这次使用F骰子,那么下次仍然使用F骰子的概率是0.95,换用L骰子的概率是0.05。相反,如果这次使用L骰子,那么下次仍然使用L骰子的概率是0.9,换用F骰子的概率是0.1。
状态向量
上面的介绍中,使用骰子的过程就是一个简单的Markov链。令为一系列掷骰子的状态,则。F为公平骰子,L为作弊骰子。Markov链中所有可能的状态构成的集合(向量),即此处的。称为状态向量。
符号向量
上图中,不同骰子状态下,掷出1-6的概率不一样,假设在第次投掷中,使用的骰子状态是,而掷出的结果(符号)为,那么。Markov链中所有可能的结果(符号)构成的集合(向量),即此处的,称为符号向量。
转移概率
为了方便,我们记,称为从状态到状态的转移概率。比如:
转移矩阵
不同状态之间互相转移的概率可以构成一个矩阵,成为转移矩阵。比如:
F | L | |
---|---|---|
F | 0.95 | 0.05 |
L | 0.1 | 0.9 |
初始向量
有了转移概率的记号,那么的计算公式可以写成:
其中,就是状态0,实际上它并不存在,只是为了建模的方便假设它存在。从状态0到其他状态的转移概率可以称为初始概率,所有可能的初始概率构成初始向量。比如:
F | L | |
---|---|---|
0 | 0.9 | 0.1 |
发射概率
不同骰子状态下某一符号出现的概率是不一样的,以骰子为例,公平骰子和作弊骰子掷出6的概率是不一样的,不光是6,掷出1-5的概率也是不一样的:
我们将某一状态下符号出现的概率称为发射概率,记为,即
发射矩阵
不同状态下不同符号的发射概率可以构成一个矩阵,成为发射矩阵,如:
1 | 2 | 3 | 4 | 5 | 6 | |
---|---|---|---|---|---|---|
F | 1/6 | 1/6 | 1/6 | 1/6 | 1/6 | 1/6 |
L | 0.1 | 0.1 | 0.1 | 0.1 | 0.1 | 0.5 |
隐马尔可夫模型
什么叫隐马尔可夫模型呢?还以上图为例。假设别人一共投掷了300次,那么我们可以看到一个长度为300的符号序列,其中。但是我们并不知道每一次投掷用的是F骰子还是L骰子,也就是说我们不知道每一次骰子的状态。这种符号序列已知,而状态序列被隐藏的模型就叫隐马尔可夫模型。
为了后续深入学习隐马模型,我们首先得写一个程序,能根据转移矩阵以及发射矩阵生成一个随机的状态序列以及相应的符号序列。
代码如下:
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
typedef char State;
typedef char Result;
State state[] = {'F', 'L'}; // 所有的可能状态
Result result[] = {'1', '2', '3', '4', '5', '6'}; // 所有的可能符号
double init[] = {0.9, 0.1}; // 初始状态的概率向量
double emission[][6] = { // 发射矩阵:行对应着状态,列对应着符号
1.0/6, 1.0/6, 1.0/6, 1.0/6, 1.0/6, 1.0/6,
0.1, 0.1, 0.1, 0.1, 0.1, 0.5
};
double trans[][2] = { // 转移矩阵:行和列都是状态
0.95, 0.05,
0.1, 0.9
};
const int nstate = 2;
const int nresult = 6;
State* rst; // 一串随机状态序列
Result* rres; // 一串随机符号序列
int random(double* prob, const int n);
void randSeq(State* st, Result* res, const int n);
void printState(State* st, const int n);
void printResult(Result* res, const int n);
int main(void) {
int i;
int n = 300;
int ll = 60;
int nl, nd;
if ((rst = (State*) malloc(sizeof(State) * (n))) == NULL || \
(rres = (Result*) malloc(sizeof(Result) * (n))) == NULL) {
fputs("Error: out of space!\n", stderr);
exit(1);
}
randSeq(rst, rres, n);
nl = n / ll;
nd = n % ll;
for (i = 0; i < nl; i++) {
printf("Rolls\t");
printResult(rres + ll * i, ll);
printf("Die\t");
printState(rst + ll * i, ll);
printf("\n");
}
if (nd > 0) {
printf("Rolls\t");
printResult(rres + ll * i, nd);
printf("Die\t");
printState(rst + ll * i, nd);
printf("\n");
}
free(rst);
free(rres);
}
// 根据一个概率向量从0到n-1随机抽取一个数
int random(double* prob, const int n) {
int i;
double p = rand() / 1.0 / (RAND_MAX + 1);
for (i = 0; i < n - 1; i++) {
if (p <= prob[i])
break;
p -= prob[i];
}
return i;
}
// 根据转移矩阵和发射矩阵生成一串随机状态和符号
void randSeq(State* st, Result* res, const int n) {
int i, ls, lr;
srand((unsigned int) time(NULL));
ls = random(init, nstate);
lr = random(emission[ls], nresult);
st[0] = state[ls];
res[0] = result[lr];
for (i = 1; i < n; i++) {
ls = random(trans[ls], nstate);
lr = random(emission[ls], nresult);
st[i] = state[ls];
res[i] = result[lr];
}
}
void printState(State* st, const int n) {
int i;
for (i = 0; i < n; i++)
printf("%c", st[i]);
printf("\n");
}
void printResult(Result* res, const int n) {
int i;
for (i = 0; i < n; i++)
printf("%c", res[i]);
printf("\n");
}
(公众号:生信了)