LabVIEW

cancel
Showing results for 
Search instead for 
Did you mean: 

Simple peak detection without threshold

Peak detection is a complex topic, but there's a simple solution if you simply need to automate finding peaks and valleys. If you simply define a peak as lying between valleys, you can use peakdet.vi, a port of a popular MATLAB script.

 

The LabVIEW code is available on my github, and attached here with a demo in LV2012. 

 

Feedback is welcome.

 

littlesphaeroid_0-1743093921917.pnglittlesphaeroid_1-1743093965960.png

 

_____________
Creator of the BundleMagic plugin for LabVIEW!
Download All
Message 1 of 11
(442 Views)

While it seems to work, there are quite a few things I would change (and I haven't even looked at the FOR loop code!)

 

  • Why do all the calculations execute even if an error occurs?
  • Place the loop code in the false case and handle errors is the true case
  • No, local variables don't simplify the code if done right and a sequence structure just clutters the diagram
  • Reading delta from a local variable inside the loop eliminates many compiler optimizations, because the compiler must assume that it can change at any time.
  • One error popup is sufficient, even if two different errors occur.
  • etc.

 

Here's how it could look like. Arguably less clutter:

 

altenbach_0-1743099059921.png

 

Message 2 of 11
(416 Views)

Yes, it is a good method, and often much faster and more flexible than the native peak finding routine in LabVIEW which you have to run twice if you want both peaks and valleys.   

 

I believe I've posted my LabVIEW code on this a few years ago and even credited the idea to the original author Eli Bilauer ( LINK: http://billauer.co.il/peakdet.html )

 

I'll be curious to see how much faster Altenbach's code is than mine! 😉

 

Craig

 

peakdet_vi_image.png

 

Message 3 of 11
(391 Views)

@littlesphaeroid wrote:

Peak detection is a complex topic, but there's a simple solution if you simply need to automate finding peaks and valleys. If you simply define a peak as lying between valleys, you can use peakdet.vi, a port of a popular MATLAB script.

 

The LabVIEW code is available on my github, and attached here with a demo in LV2012. 

 

Feedback is welcome.

 

littlesphaeroid_0-1743093921917.pnglittlesphaeroid_1-1743093965960.png

 


Thanks Russell. I remember using this script with Matlab in the past.

 

One question - The algorithm depends on a delta between the valleys and peaks, why do you define this as "without threshold"? Isn't that delta the threshold?

0 Kudos
Message 4 of 11
(383 Views)

@mcduff wrote:
One question - The algorithm depends on a delta between the valleys and peaks, why do you define this as "without threshold"? Isn't that delta the threshold?

Its a "relative" threshold (applied to the first derivative of the signal, not the signal itself) and the peak detection depends on two factors: (1) The relative difference between two adjacent points AND next point lower than second. I have the nagging feeling that this could be implemented with 10% of the current code. 😄

 

Quote from the text code comment:

 

% A point is considered a maximum peak if it has the maximal
% value, and was preceded (to the left) by a value lower by
% DELTA

 

Personally, this feels more like a "spike detector", because e.g. a beautifully smooth and wide gaussian peak would not be detected as such.

Message 5 of 11
(330 Views)

@cstorey wrote: Yes, it is a good method, and often much faster and more flexible than the native peak finding routine in LabVIEW which you have to run twice if you want both peaks and valleys.   

 

Craig, I'm sorry I didn't find your port of this code sooner. I hope it is noticed that I also fully attributed the source of the code. 

 

@mcduff wrote:
One question - The algorithm depends on a delta between the valleys and peaks, why do you define this as "without threshold"? Isn't that delta the threshold?

 

In LabVIEW's built-in peak finding methods, the threshold is an absolute value, which may require a relatively flat baseline, and often a fair amount of signal conditioning. I wanted to include threshold in the title to help people find this code, which uses a "threshold" (here "delta") which is relative to the baseline, or more precisely the nearest valley. I admit it's difficult to find suitable language to make this distinction.

 

@altenbach wrote:

Personally, this feels more like a "spike detector", because e.g. a beautifully smooth and wide gaussian peak would not be detected as such.

 

a) A useless semantic distinction, and b) incorrect. On the latter point, there is no practical difference between a  "broad" and some other gaussian, and peakdet is by design agnostic to noise ("smoothness"). But importantly, LabVIEW isn't a "feels" language. A simple test shows a Gaussian of any width is detected as long as it exceeds the delta value (as shown in the demo provided, which contains a trace with multiple gaussians).

 

In my use case the data acquisition time is ~5 orders longer than the analysis. Running a 100kS data set through takes 2-5ms on my laptop. Improving its performance would not improve my application in any meaningful way, though I can see several routes to doing so. Have at big guy.

 

 

For the community, I have prepared a version without the initial error checking, in keeping with other LV analysis methods. I hope people find it and find it useful!

_____________
Creator of the BundleMagic plugin for LabVIEW!
Message 6 of 11
(256 Views)

@littlesphaeroid wrote:

 

@altenbach wrote:

Personally, this feels more like a "spike detector", because e.g. a beautifully smooth and wide gaussian peak would not be detected as such.

 

a) A useless semantic distinction, and b) incorrect. On the latter point, there is no practical difference between a  "broad" and some other gaussian, and peakdet is by design agnostic to noise ("smoothness"). But importantly, LabVIEW isn't a "feels" language. A simple test shows a Gaussian of any width is detected as long as it exceeds the delta value (as shown in the demo provided, which contains a trace with multiple gaussians)..


As demonstration, here are your examples with 5% Gaussian noise added: (and no, even tuning delta by an order of magnitude in either direction does not change the outcome).

 

altenbach_0-1743435370381.png

 

 

Since delta depends on a first derivative and not on peak amplitude it is difficult to guess a good value without trying. A broad gaussian has nearly zero slope around the peak and any noise will overshadow the delta.

 

altenbach_0-1743433160571.png

altenbach_2-1743433962703.png

altenbach_3-1743433998117.png

 

 

 

altenbach_1-1743433272323.png

 

 


@littlesphaeroid wrote:

In my use case the data acquisition time is ~5 orders longer than the analysis. Running a 100kS data set through takes 2-5ms on my laptop. Improving its performance would not improve my application in any meaningful way, though I can see several routes to doing so. Have at big guy.

 


I was not talking about "run performance", just by the simple fact that if an algorithm can be explained in one sentence, it should not need a page full of complicated code. I am sure it could be done with 10%, making it 10x faster to write, easier to debug and there would be many fewer places for bugs to hide.

 

It should also be mentioned that [x] (if provided) must be sorted.

Message 7 of 11
(225 Views)

@altenbach wrote:

Since delta depends on a first derivative and not on peak amplitude it is difficult to guess a good value without trying. A broad gaussian has nearly zero slope around the peak and any noise will overshadow the delta.

 

altenbach_0-1743433160571.png

 

 



I have been using the WA Multiscale Peak Detection VI in the Advanced Signal Processing Toolkit for peak finding. It works well provided - 

  1. Noisy data needs to be smoothed first, e.g., with a Savitzky-Golay Filter.
  2. The VI allows the user to detrend the data and remove the baseline if needed.
  3.  The "width" needs to be well specified. The width is the number of samples of the peaks or valleys.
  4. You would need to run twice to get both peaks and valleys.

The peak fit contribution posed here is nice, small, easily accessible, but may not fit every situation. Below is the example from the Wavelet algorithm. When the signal is detrended, baseline removed works well, when not so well.

 

Baseline/Trend Removed

mcduff_1-1743438844289.png

Baseline/Trend Not Removed

mcduff_3-1743438936880.png

 

 

Message 8 of 11
(204 Views)

@mcduff wrote:

@altenbach wrote:

Since delta depends on a first derivative and not on peak amplitude it is difficult to guess a good value without trying. A broad gaussian has nearly zero slope around the peak and any noise will overshadow the delta.

Thanks mcduff, your example points out some of the difficulties with other peakfinding methods.

 

In my use case peakdet has another useful feature. I'm working on an RF regulatory compliance tool, where the max power at each peak must be compared to a limit. peakdet returns the max value for each peak, rather than some sort of fit to the peak, which may give a peak center rather than a max.  Perfect!

 

Regarding prior messages here, obviously you need to know something about the peak heights and noise levels in your signal in order to select an appropriate delta.

_____________
Creator of the BundleMagic plugin for LabVIEW!
0 Kudos
Message 9 of 11
(154 Views)

 

@altenbach wrote: some bad examples.

 

Showed provided examples not working after adding noise (to already noisy signals), used the provided default deltas, and claimed that changing the deltas did not affect the result. 

 

I'd hoped it would be obvious that providing deltas larger than the noise and smaller than the peaks would produce the desired results:

 

littlesphaeroid_0-1743603295331.png

 

littlesphaeroid_1-1743603354328.png

 

littlesphaeroid_2-1743603411925.png

 

 

_____________
Creator of the BundleMagic plugin for LabVIEW!
0 Kudos
Message 10 of 11
(104 Views)