`

Viterbi 算法应用实现

阅读更多

算法简介:

Viterbi 算法又叫维特比算法,其是HMM模型中在给出观测序列O,以及状态转移举证M 和 状态-观测矩阵Q之后,求解能够最佳解释序列O的状态序列S的一种动态规划算法。

具体如下图所示:

其中:

 

  •   标记为O的 [0|1] 序列是观测序列, 
  •   标记为S的序列中横向的箭头即状态在根据转移矩阵M进行转移,
  •   其中S序列与O序列之间向下的箭头表示根据状态生成观测的概率,也即Q矩阵。

假设我们已知观测序列O,即图片Bottom位置的0、1序列,则图片Top位置的用红框框起来的0、1序列就是对应的S序列,即我们所说的在给定矩阵M和Q之后,能最佳地解释O的序列。

 

算法实现:

算法的实现也很简单,主要包含两步:

1. 从前到后遍历O序列,根据观测生成的方式,计算每个观测的最大生成概率。

    在计算过程中,当前观测只依赖于上一个观测的结果,即马氏性。

    同时记录当前状态最有可能的上一个状态。

2. 从后向前回溯,得到序列S

    根据1中在每个观测点记录的上一个最佳状态进行回溯。

 

具体实现如下:

/* ========================================================
 *   Copyright (C) 2014 All rights reserved.
 *   
 *   filename : viterbi.c
 *   author   : ***
 *   date     : 2014-12-19
 *   info     : 
 * ======================================================== */

#include <math.h>
#include <string.h>
#include "viterbi.h"

/* ****************************************
 * @param l   : input 0|1 vector 
 * @param o   : output[smoothed] 0|1 vector
 * @param n   : vector length
 * @param r_t : rate for stat <--> stat
 * @param r_o : rate for stat <--> observ
 * ****************************************/
int viterbi(int * l, int * o, int n, int r_t, int r_o){
    // check params 
    if (!l || !o || r_t < 1 || r_o < 1){
        // return fail
        return -1;
    }

    int i = 0;
    double r_t_ = log(r_t);
    double r_o_ = log(r_o);
    double ls0 = 0.0, ls1 = 0.0, cs0 = 0.0, cs1 = 0.0;
    double ps0 = 0.0, ps1 = 0.0;

    // prev_state for trace back
    int (*pre_s)[2] = (int(*)[2])malloc(sizeof(int[2]) * n);

    // init pre_state and output vector
    memset(pre_s, 0, sizeof(int[2]) * n);
    memset(o, 0, sizeof(int) * n);

    // init the begin stat distribute
    ls0 = (l[0] == 0 ? r_o_ : 0.0);
    ls1 = (l[0] == 1 ? r_o_ : 0.0);

    // pass through to calculate max prob
    for (i = 1; i < n ; i++){
        ps0 = ls0 + r_t_ + (l[i] == 0 ? r_o_ : 0.0);
        ps1 = ls1 +        (l[i] == 0 ? r_o_ : 0.0);
        if (ps0 >= ps1){
            cs0 = ps0; pre_s[i][0] = 0;
        }
        else{
            cs0 = ps1; pre_s[i][0] = 1;
        }
        ps0 = ls0 +        (l[i] == 1 ? r_o_ : 0.0);
        ps1 = ls1 + r_t_ + (l[i] == 1 ? r_o_ : 0.0);
        if (ps0 >= ps1){
            cs1 = ps0; pre_s[i][1] = 0;
        }
        else{
            cs1 = ps1; pre_s[i][1] = 1;
        }
        ls0 = cs0; ls1 = cs1;
    }

    // end state with the max prob
    o[n - 1] = cs0 >= cs1 ? 0 : 1;

    // trace back for prev state
    for (i = n - 2; i >= 0; i--){
        o[i] = pre_s[i + 1][o[i + 1]];
    }

    // free the trace pre states
    free(pre_s);
    pre_s = NULL;
    
    // return success
    return 0;
}

 

测试代码如下:

int main(){
    int a[15] = {0,0,1,0,0,1,1,1,0,1,1,0,0,0,0};
    int b[15] = {0,0,1,0,0,1,1,1,0,1,1,0,0,0,0};
    int c = viterbi(a,b,15,10,5);
    int i = 0;
    printf("before smooth:\n");
    for (; i < 15; i++){
        printf("%d ", a[i]);
    }
    printf("\n");
    printf("after smooth:\n");
    for (i = 0; i < 15; i++){
        printf("%d ", b[i]);
    }
    printf("\n");
}

 

结果如下:

before smooth:
0 0 1 0 0 1 1 1 0 1 1 0 0 0 0 
after smooth:
0 0 0 0 0 1 1 1 1 1 1 0 0 0 0 

 

结果中显示原始序列中第3位置的“1”被平滑成“0”,原始序列中第9位置的“0”被平滑成"1"。

 

该算法在处理时序异常检测结果中的异常序列十分有用。

可以将平滑后的连续“1” 看成一个异常事件,而不用针对每个检测到的异常点单独处理。

  • 大小: 14.8 KB
1
0
分享到:
评论
2 楼 liuzhiqiangruc 2015-06-03  
.h 文件里就一行:
就是一个函数申明而已
int viterbi(int * l, int * o, int n, int r_t, int r_o);
1 楼 yifangt 2015-06-03  
Where is your viterbi.h file, please?

相关推荐

    基于Viterbi-Viterbi的载波相位估计算法

    Viterbi-Viterbi算法(也称为联合检测或最大似然序列估计)是一种优化的实现方法,它结合了Viterbi算法的基本思想来解决载波相位的估计问题。 Viterbi算法最初是为了解码分组码而设计的,它通过找到最可能的码序列...

    Viterbi算法与实现 注释版

    总的来说,Viterbi算法是信息论和统计建模中的一个重要工具,它在处理序列数据和找出最可能解释这些数据的隐藏状态序列时有着广泛的应用。通过理解其原理和实现,我们可以解决各种领域中的复杂问题,从通信系统的...

    HMM前/后向算法、Viterbi算法实现

    在这个项目中,我们重点探讨了HMM的三个核心算法:前向算法、后向算法以及Viterbi算法,并使用Java编程语言进行了实现。 1. **前向算法**: 前向算法是计算HMM在给定观测序列下模型概率的动态规划方法。它通过计算...

    Viterbi算法与实现

    随着时间的发展,Viterbi算法在模式识别、自然语言处理、生物信息学以及人脸识别等领域得到了广泛应用。 在人脸识别中,Viterbi算法可以用来解析一系列的面部特征观测,以确定最有可能的一系列人脸表情或识别状态。...

    Viterbi算法c/c++实现

    Viterbi算法是一种在通信、信息处理和计算机科学领域广泛应用的动态规划算法,主要用于序列概率模型,如马尔科夫模型。在这个特定的C/C++实现中,我们聚焦于卷积码的解码过程。 卷积码是通信系统中常用的前向错误...

    基于FPGA的Viterbi算法改进及其实现.pdf

    Viterbi算法是一种广泛应用于数字通信领域的译码算法,由Andrew Viterbi于1967年提出,主要用于解决差错控制编码中的最大似然序列估计问题。其核心思想是采用动态规划方法,通过比较和选择使得整个路径概率最大的...

    经典的Viterbi算法

    在给定的压缩包文件中,"Readme.txt"可能是对Viterbi算法的进一步解释或使用说明,而"viterbi"可能是一个程序实现或者示例代码,可以帮助读者更深入地理解和应用Viterbi算法。通过阅读这些文件,你可以了解如何在...

    Viterbi算法Java实例

    本篇文章通过一个Java实现案例详细介绍了Viterbi算法在隐马尔科夫模型中的应用。通过学习本文,读者可以更好地理解Viterbi算法的工作原理及其具体实现方式,为今后在相关领域的研究与开发打下坚实的基础。

    phaseCompensate_mod_v1_MOD_载波恢复_相位恢复_Viterbi-Viterbi算法_载波相位恢复_源码

    mod_v1_MOD_载波恢复_相位恢复_Viterbi-Viterbi算法_载波相位恢复_源码"的压缩包,其中包含了一个名为`phaseCompensate_mod_v1.m`的MATLAB源代码文件,该文件实现了基于Viterbi-Viterbi算法的相位恢复过程。...

    隐马尔可夫模型中的Viterbi算法zz

    在给定的文件"Viterbi.java"中,很可能包含了Viterbi算法的Java实现。通常,这个类可能会包含初始化概率数组,动态规划计算,以及回溯路径的函数。通过阅读源码,我们可以更深入地理解算法的细节,包括如何处理边界...

    viterbi算法

    Viterbi算法是一种在通信和信息处理领域广泛应用的序列概率最大似然解码方法,尤其在数字通信、信息编码和模式识别中具有重要地位。它最初由Andrew Viterbi在1967年提出,主要用于纠正传输过程中的错误,确保信息的...

    freqCompensate_载波恢复_频偏_Viterbi-Viterbi算法_频偏恢复_

    Viterbi-Viterbi算法因其高效性和准确性在现代通信系统中得到广泛应用,尤其是在数字卫星通信、无线通信以及光纤通信等领域。通过理解和掌握这一算法,工程师们可以设计出更稳健的接收机,提高通信系统的整体性能。...

    Viterbi算法实现

    Viterbi算法是一种在隐马尔可夫模型(Hidden Markov Model, HMM)中寻找最可能序列的动态规划算法,常被应用于模式识别、语音识别、自然语言处理和序列数据分析等多个领域。在这个C#实现中,我们主要探讨Viterbi算法...

    freqCompensate_载波恢复_频偏_Viterbi-Viterbi算法_频偏恢复_源码.zip

    在实际应用中,Viterbi-Viterbi算法通常与其他技术(如Costas环、PLL或M-ary序列同步)结合使用,以实现更高效的载波恢复。源代码分析可以帮助我们理解如何在实际系统中实现这些复杂算法,并优化通信性能。 总之,...

    viterbi算法.pdf

    特别是在通信领域,比如在用于解码纠错码和信道编码,以及在语音识别和自然语言处理中,Viterbi算法也有着广泛的应用。 在理解和应用Viterbi算法时,有几个关键概念需要掌握: - 隐状态(Hidden States):模型中不...

    viterbi算法linux下C++实现.docx

    ### Viterbi算法在Linux下的C++实现 #### 一、引言 Viterbi算法是一种动态规划算法,主要用于隐马尔可夫模型(HMM)中的最优化问题,特别是序列解码问题。它能有效地找到观察序列中最有可能的隐藏状态序列。在语音...

    viterbi译码算法c++实现以及程序所对应的转移图

    "Viterbi译码算法C++实现及程序对应的转移图" 在这篇文章中,我们将详细讲解Viterbi译码算法的实现细节,并...本文提供了Viterbi译码算法的C++实现代码和详细的讲解,旨在帮助读者更好地理解和应用Viterbi译码算法。

    c++的viterbi算法

    Viterbi算法是一种在概率模型中寻找最可能序列的动态规划方法,特别是在 Hidden Markov Model (HMM) 中广泛应用。在C++中实现Viterbi算法,我们需要理解HMM的基本概念,包括状态、观测和转移概率。 首先,让我们...

    Viterbi译码算法在55系列DSP上的C语言实现.pdf

    在本研究中,重点在于将Viterbi算法应用在TI公司生产的55系列DSP(数字信号处理器)上,使用C语言进行编程实现。在现有的技术文献中,采用C语言实现Viterbi译码算法的详细论述并不常见,很多实现都是通过汇编语言...

    基于Viterbi算法以及预训练模型用于中文分词标注功能实现

    本文将深入探讨如何结合Viterbi算法与预训练模型来实现中文分词标注功能。 首先,我们需要理解Viterbi算法。这是一种动态规划方法,常用于寻找最有可能的序列状态,如在隐马尔科夫模型(HMM)中找到最可能的词序列...

Global site tag (gtag.js) - Google Analytics