滚动平均值,vb.net

时间:2016-10-17 12:15:20

标签: vb.net average transducer

我有一个传感器(实际上是一个Quartzonix压力传感器),通过串口吐出数据,大约每秒3次。我想设置一些代码,根据x样本量给出平均读数。

输出看起来像这样:

01+   1.502347091823e01
01+   1.501987234092e01
01+   1.50234524524e01
01+   1.502123412341e01
01+   1.502236234523e01
01+   1.50198345e01
01+   1.502346234523e01

..并将一直持续到com端口关闭或传感器获得另一个命令。

这是我到目前为止的代码,代码可以告诉我传感器实际输出的是什么:

Private Sub btnStart_Click(sender As Object, e As EventArgs) Handles btnStart.Click
    Dim a As String
    a = "$01MC" & Chr(13)
    MyComPort.WriteLine(a)

    Do
        Dim Incoming As String = MyComPort.ReadLine()
        Dim incomingtext As String = Incoming.Remove(0, 3)

        If Incoming Is Nothing Then
            Exit Do
        Else
            txtRawData.Text = Incoming
            boxPSIA.Text = Format(Val(incomingtext), "##0.000")

        End If
        Application.DoEvents()
    Loop

End Sub

“$ 01MC”是传感器开始吐出数据所需的命令。当我点击开始按钮时,我发生了一些奇怪的超时事件,但这是另一个节目(可能需要调整.readtimeout,不确定)。

我有一个文本框txtReadingsToAvg,用于输入平均读数的数量。我只是不知道如何实际让它来计算平均值(例如,按一下按钮然后吐出它)进入msgbox,甚至进入另一个文本框)。

1 个答案:

答案 0 :(得分:1)

不确定您的代码是如何工作的。你说你得到的值约为3赫兹?然后直线Do...Loop太快了。当串行端口接收数据时会引发一个事件。利用它。

您可能需要稍微更改一下以满足您的需求

' WithEvents allows events to be handled with "Handles" keyword
Private WithEvents myComPort As SerialPort
Private dataQueue As Queue(Of Double)
Private numReadingsToAvg As Integer = 0

Private Sub btnStart_Click(sender As Object, e As EventArgs) Handles btnStart.Click
    ' make a new queue here to initialize or clear an old queue
    dataQueue = New Queue(Of Double)()
    ' read the num avgs text box. you may want to change on the fly also
    numReadingsToAvg = Integer.Parse(Me.txtReadingsToAvg.Text)
    myComPort.WriteLine("$01MC" & Chr(13))
End Sub

Private Sub myComPort_DataReceived(sender As Object, e As SerialDataReceivedEventArgs) Handles myComPort.DataReceived
    Dim incomingLine As String = myComPort.ReadLine()
    ' DataReceived event happens on its own thread, not the UI
    ' must invoke call back to UI to change properties of controls
    txtRawData.Invoke(Sub() txtRawData.Text = incomingLine)
    Dim incomingValue As String = incomingLine.Remove(0, 3).Trim()
    If Not String.IsNullOrWhiteSpace(incomingValue) Then
        Exit Sub
    Else
        Dim measurement As Double = Double.Parse(incomingValue)
        boxPSIA.Invoke(Sub() boxPSIA.Text = Format(measurement, "##0.000"))
        dataQueue.Enqueue(measurement)
        ' if there are too many items, remove the last one
        If dataQueue.Count > numReadingsToAvg Then
            dataQueue.Dequeue()
        End If
        Dim average As Double = dataQueue.Average()
        ' you need to add this textbox
        anotherTextBox.Invoke(Sub() anotherTextBox.Text = Format(average, "##0.000"))
    End If
End Sub

顺便说一句,Application.DoEvents() should rarely (never) be used, as there's always a better way to remedy whatever problem you are bandaging with DoEvents。您的原始示例在UI上运行了一个永无止境的循环,从而阻塞了UI线程。如果您需要运行这样的循环,它应该几乎总是在与UI线程不同的线程上运行。在我的例子中,没有循环,时间由端口本身决定。不需要在UI线程上发生任何此类事件。