加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

数组 – 如何知道数组是否代表正弦波?

发布时间:2020-12-14 04:46:50 所属栏目:百科 来源:网络整理
导读:我正在使用加速度计并在过去的几秒钟内收集数据.我想要检测的运动可以用运动传感器得到的值表示为正弦波.所以为了确保,我想要一种方法来检查从传感器返回的数据是否代表正弦波. 我想避免的是手动比较数组中的每个值并做出决定. 我想知道是否有一种有效的方法
我正在使用加速度计并在过去的几秒钟内收集数据.我想要检测的运动可以用运动传感器得到的值表示为正弦波.所以为了确保,我想要一种方法来检查从传感器返回的数据是否代表正弦波.

我想避免的是手动比较数组中的每个值并做出决定.

我想知道是否有一种有效的方法可以判断我的数组是否代表正弦波.

解决方法

正如评论员@NeilForrester所指出的那样,FFTs就是这样做的.编写自己的高效FFT并不容易,但是如果使用Objective-C,Accelerate框架的 vDSP routines提供了一种直接的方法 – 由于使用了UnsafePointer和UnsafeMutablePointer参数,因此在Swift中并不那么简单.这是一个使用FFT的简单Swift示例.

import Foundation
import Accelerate

public struct GFFT {
    let size: Int
    let halfSize: Int
    let log2n: vDSP_Length
    let twoOverSize: [Float]
    var weights: FFTSetup

    init?(size: Int) {
        self.size = size
        self.log2n = vDSP_Length(log2(Float(size)))
        guard let weights = vDSP_create_fftsetup(log2n,FFTRadix(kFFTRadix2)) else {
            print("Aargh in GFFT.fft - weights failed")
            return nil
        }
        self.halfSize = size / 2
        self.twoOverSize = [2 / Float(size)]
        self.weights = weights
    }

    public func forward(realArray: [Float]) -> (magnitude: [Float],phase: [Float]) {
        assert(realArray.count == self.size,"Aargh in GFFT.forward - size mismatch")
        var real = realArray // copy into var
        var imag = GFFT.zeros(size)
        var magnitudesSquared = GFFT.zeros(self.halfSize)
        var magnitudes = GFFT.zeros(self.halfSize)
        var normalizedMagnitudes = GFFT.zeros(self.halfSize)
        var phases = GFFT.zeros(self.halfSize)

        var splitComplex = DSPSplitComplex(realp: &real,imagp: &imag)

        vDSP_fft_zip(self.weights,&splitComplex,1,self.log2n,FFTDirection(FFT_FORWARD))

        vDSP_zvmags(&splitComplex,&magnitudesSquared,vDSP_Length(self.halfSize))
        vvsqrtf(&magnitudes,[Int32(self.halfSize)])

        vDSP_zvphas(&splitComplex,&phases,vDSP_Length(self.halfSize))

        vDSP_vsmul(&magnitudes,self.twoOverSize,&normalizedMagnitudes,vDSP_Length(self.halfSize))

        // you may choose to return magnitudesSquared,for the power
        // magnitudes for the scaled amplitudes or 
        // normalizedMagnitudes for,well,normalised magnitude.
        return (normalizedMagnitudes,phases)
    }
    private static func zeros(_ n: Int) -> [Float] { return [Float](repeating: 0,count: n) }
}

let testInput = (0 ..< 512).map {
    return sin(Float($0))
}
if let fft = GFFT(size: testInput.count) {
    let (freq,phase) = fft.forward(realArray: testInput)
    freq.map({$0})
}

游乐场输出:

FFT graph

至于你测试什么,它将取决于你得到的实际输出,所以我会试验实际数据给你的东西,但你的测试应该是这样的:

>找到振幅的平均值>找到最大振幅>检查两者的比率(最大值/平均值)是否高>检查最大值的索引是否接近零(或者它可能是直流信号)>检查任何其他局部最大值(并且会有一些)远小于全局最大值.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读