核心寻峰算法的原理参考Ronny,链接:投影曲线的波峰查找

C#翻译原理代码参考sowhat4999,链接:C#翻译Matlab中findpeaks方法

前人种树,后人乘凉。感谢原作者详细的解释说明。

这里先把翻译代码贴一下(略微的修改了sowhat4999代码中的几个参数)

//调用方法
List<double> data = new List<double>{, , , , , , , , , , };
List<int> index = getPeaksIndex(trendSign(oneDiff(data)));

//第一次寻峰(基本峰距为1)算法
private double[] oneDiff(List<double> data)
{
double[] result = new double[data.Count - ];
for (int i = ; i < result.Length; i++)
{
result[i] = data[i + ] - data[i];
}
return result;
} private int[] trendSign(double[] data)
{
int[] sign = new int[data.Length];
for (int i = ; i < sign.Length; i++)
{
if (data[i] > ) sign[i] = ;
else if (data[i] == ) sign[i] = ;
else sign[i] = -;
} for (int i = sign.Length - ; i >= ; i--)
{
if (sign[i] == && i == sign.Length - )
{
sign[i] = ;
}
else if (sign[i] == )
{
if (sign[i + ] >= )
{
sign[i] = ;
}
else
{
sign[i] = -;
}
}
}
return sign;
} private List<int> getPeaksIndex(int[] diff)
{
List<int> data = new List<int>();
for (int i = ; i != diff.Length - ; i++)
{
if (diff[i + ] - diff[i] == -)
{
data.Add(i + );
}
}
return data;//相当于原数组的下标
}

以上方法并没有将峰距、边锋、峰值情况考虑在内,但已经给与我们后人一个完整的思路。

峰距情况分析:

我们可以将上述方法理解为峰距1的寻峰算法,当我们需要完成峰距为2的寻峰情况时我们需要判断

data[i]是否大于data[i+1],data[i+2],data[i-1],data[i-2]

同理按照此方法完成点数为100000,峰距为1000的寻峰,则需要进行100000的1000次方次运算,这显然需要花费大量的时间进行运算。

优化过程中,我们并不能改变峰距(即幂指数1000),但我们可以改变点数(即底数100000)的大小。从而实现运算量的降低。

以上峰距为1的寻峰方法此时已经完成判断

data[i]是否大于data[i+1],data[i-1]

并返还峰值对应的索引列

峰距为2时,我们只需要再次对索引列中内容进行判断即可(只有在峰距为1的判断中胜出的点,才有可能在峰距为2的判断中胜出)

data[i]是否大于data[i+2],data[i-2]

此时你会发现我们需要遍历的底数已经并不是原点数100000,而是上次返还的寻峰序列个数

            // 调用方法
List<double> Xaxis = new List<double> { , , , , , , , , , , };
List<double> Yaxis = new List<double> { , , , , , , , , , , };
// 峰距
int DisPeak = ;
// 峰距为3时得到的脚标
List<int> index = getPeaksIndex(trendSign(oneDiff(Yaxis)));
// 已进行的判断
int level = ;
// 扩大峰距范围范围算法
while (DisPeak > level)
{
level++;
List<int> result = DoPeakInstance(Yaxis, index, level);
index = null;
index = result;
} // 获取两侧满足条件的边峰序列
index = GetBothSidePeakIndex(Xaxis, Yaxis, , index); double minFZ = 10.0;
// 根据最小峰值序列进行筛选
index = FindMinPeakValue(minFZ, Yaxis, index);
        //扩大寻峰范围算法
private List<int> DoPeakInstance(List<double> data, List<int> index, int level)
{
//相当于原数组的下标
List<int> result = new List<int>();
for (int i = ; i < index.Count; i++)
{
//判断是否超出下界和上界
if (index[i] - level >= && index[i] + level < data.Count)
{
if (data[index[i] + level] <= data[index[i]] && data[index[i] - level] <= data[index[i]])
{
result.Add(index[i]);
}
}
}
return result;
}

边锋情况分析:

仔细阅读上述两算法,你会发现该算法存在一个无法避免的问题 如:

峰距是3,此时峰首部点序(点0,点1,点2)因无法向前比较,导致并没有参与到峰值计算中。 尾部点则因无法向后比较没有参与到峰值计算中。

此情况我们首先要清楚,因上述情况未参与比较的点序中,首部最多仅有一个峰值,尾部最多仅有一个峰值。

那我们把它加上就好了,美滋滋。

        //获取两侧满足条件的边峰序列
private static List<int> GetBothSidePeakIndex(List<double> Xaxis, List<double> Yaxis, int FJ, List<int> index)
{
//获取数据首尾两侧最大峰值(0,FJ)点序和(Date.CountFJ-FJ,Data.Count)点序
int TopIndex = ;
int BottomIndex = Yaxis.Count - ;
for (int i = ; i < FJ; i++)
{
if (Yaxis[i] >= Yaxis[TopIndex])
{
TopIndex = i;
}
if (Yaxis[Yaxis.Count - - i] >= Yaxis[BottomIndex])
{
BottomIndex = Yaxis.Count - - i;
}
}
//判断是否满足条件检索条件
int newTopIndex = TopIndex;
int newBottomIndex = BottomIndex;
for (int i = ; i <= FJ; i++)
{ if (Yaxis[TopIndex + i] >= Yaxis[TopIndex])
{
newTopIndex = TopIndex + i;
}
if (Yaxis[BottomIndex - i] >= Yaxis[BottomIndex])
{
newBottomIndex = BottomIndex - i;
}
}
TopIndex = newTopIndex;
BottomIndex = newBottomIndex; //添加到结果序列
if (TopIndex <= FJ && TopIndex != )
{
index.Insert(, TopIndex);
}
if (BottomIndex >= BottomIndex - FJ && BottomIndex != Xaxis.Count - )
{
index.Add(BottomIndex);
}
return index;
}

最后,也就是最简单的峰值判断了。比一下就好了。

        //根据最小峰值序列进行筛选
private static List<int> FindMinPeakValue(double minFZ, List<double> Yaxis, List<int> index)
{
List<int> finalresult = new List<int>();
for (int i = ; i < index.Count; i++)
{
if (Yaxis[index[i]] >= minFZ)
{
finalresult.Add(index[i]);
}
}
index = null;
index = finalresult;
return index;
}

C# 实现寻峰算法的简单优化(包含边峰,最小峰值,峰距)的更多相关文章

  1. 机器学习&数据挖掘笔记_16(常见面试之机器学习算法思想简单梳理)

    前言: 找工作时(IT行业),除了常见的软件开发以外,机器学习岗位也可以当作是一个选择,不少计算机方向的研究生都会接触这个,如果你的研究方向是机器学习/数据挖掘之类,且又对其非常感兴趣的话,可以考虑考 ...

  2. paper 17 : 机器学习算法思想简单梳理

    前言: 本文总结的常见机器学习算法(主要是一些常规分类器)大概流程和主要思想. 朴素贝叶斯: 有以下几个地方需要注意: 1. 如果给出的特征向量长度可能不同,这是需要归一化为通长度的向量(这里以文本分 ...

  3. 机器学习&数据挖掘笔记(常见面试之机器学习算法思想简单梳理)

    机器学习&数据挖掘笔记_16(常见面试之机器学习算法思想简单梳理) 作者:tornadomeet 出处:http://www.cnblogs.com/tornadomeet 前言: 找工作时( ...

  4. [转]机器学习&数据挖掘笔记_16(常见面试之机器学习算法思想简单梳理)

    机器学习&数据挖掘笔记_16(常见面试之机器学习算法思想简单梳理) 转自http://www.cnblogs.com/tornadomeet/p/3395593.html 前言: 找工作时(I ...

  5. SSE图像算法优化系列二:高斯模糊算法的全面优化过程分享(一)。

    这里的高斯模糊采用的是论文<Recursive implementation of the Gaussian filter>里描述的递归算法. 仔细观察和理解上述公式,在forward过程 ...

  6. 双数组trie树的基本构造及简单优化

    一 基本构造 Trie树是搜索树的一种,来自英文单词"Retrieval"的简写,可以建立有效的数据检索组织结构,是中文匹配分词算法中词典的一种常见实现.它本质上是一个确定的有限状 ...

  7. kylin简单优化cube

    优化Cube 层次结构 理论上,对于N维,你最终会得到2 ^ N维组合.但是对于某些维度组,不需要创建这么多组合.例如,如果您有三个维度:洲,国家,城市(在层次结构中,“更大”维度首先出现).在深入分 ...

  8. mysql简单优化思路

    mysql简单优化思路 作为开发人员,数据库知识掌握的可能不是很深入,但是一些基本的技能还是要有时间学习一下的.作为一个数据库菜鸟,厚着脸皮来总结一下 mysql 的基本的不能再基本的优化方法. 为了 ...

  9. Java 排序算法-冒泡排序及其优化

    Java 排序算法-冒泡排序及其优化 什么是冒泡排序 基本写法 优化后写法 终极版本 源码及测试 什么是冒泡排序 这里引用一下百度百科上的定义: 冒泡排序(Bubble Sort),是一种计算机科学领 ...

随机推荐

  1. 程序媛计划——python数据库

    #实例:用数据库存储日记,实现日记本功能 #流程 #创建数据库 #coding:utf-8 import sqlite3 connect=sqlite3.connect('test.db') conn ...

  2. jzoj5888

    tj:暴力連邊會tle 我們發現所有邊的邊權最大值不超過100000,這意味著可以設計和邊權有關的算法,假設現在邊權不相同 枚舉一個現在的邊權i,代表gcd為i,設連的2個點權值為a1∗ia1*ia1 ...

  3. Recursion-687. Longest Univalue Path

    Given a binary tree, find the length of the longest path where each node in the path has the same va ...

  4. MySQL(视图、触发器、函数)

    day61 参考:http://www.cnblogs.com/wupeiqi/articles/5713323.html 视图 视图:给某个查询语句设置别名,日后方便使用               ...

  5. 内核格式化(C++)

    参考<C++ Primer Plus>P788 iostream族支持 程序 与 终端 之间的I/O fstream族支持 程序 与 文件 之间的I/O sstream族支持 程序 与 s ...

  6. 输出图中顶点i到顶点j之间的所有简单路径

    简单路径(不包括环) DFS遍历以及回溯得到结果 void dfs(ALGraph graph, int v, int end, bool visit[], int path[], int cnt) ...

  7. spring boot 使用application.properties 进行外部配置

    application.properties大家都不陌生,我们在开发的时候,经常使用它来配置一些可以手动修改而且不用编译的变量,这样的作用在于,打成war包或者jar用于生产环境时,我们可以手动修改环 ...

  8. Android多媒体整体架构图

    Android多媒体整体架构图 MediaPlayer框架图 Camera框架图 SoundRecorder框架图 VideoCamera框架图 OpenCore与Skia ALSA Audio框架图 ...

  9. dataTable 参数说明

    下面是一些常用的参数列表,比较常用或者有价值的标示为绿色. 功能参数(Features) 参数名 说明 参考值 默认值 autoWidth 定义是否由控件自动控制列宽 Boolean true def ...

  10. 【bzoj4240】 有趣的家庭菜园 树状数组

    这一题最终要构造的序列显然是一个单峰序列 首先有一个结论:一个序列通过交换相邻的元素,进行排序,最少的交换次数为该序列的逆序对个数 (该结论很久之前打表意外发现的,没想到用上了.....) 考虑如何构 ...