繁体   English   中英

C#应用程序:从音频输出采样音频-> FFT算法->可视化

[英]C# Application: Sample Audio from Audio Output -> FFT Algorithm -> Visualize

我是音频新手,并且第一次使用FFT。 我想从音频输出中采样音频。 之后,我想使用FFT算法计算此数据。 我正在使用Naudio.dll

采样要求:

  • 音频输出样本
  • 采样频率必须在GUI上可调
  • 缓冲区大小必须在GUI上可调
  • 振幅值必须是原始值(无对数滤波器/无sqrt()滤波器...)

我怎么解决这个问题? 使用哪个dll?

我尝试使用NAudio的示例聚合器。 但是我不知道如何。

提前致谢

public class SampleAggregator : ISampleProvider
{
    public event EventHandler<MaxSampleEventArgs> MaximumCalculated;
    private float maxValue;
    private float minValue;
    public int NotificationCount { get; set; }
    int count;

    public event EventHandler<FftEventArgs> FftCalculated;
    public bool PerformFFT { get; set; }
    private readonly Complex[] fftBuffer;
    private readonly FftEventArgs fftArgs;
    private int fftPos;
    private readonly int fftLength;
    private readonly int m;
    private readonly ISampleProvider source;

    private readonly int channels;

    public SampleAggregator(ISampleProvider source, int fftLength = 1024)
    {
        channels = source.WaveFormat.Channels;
        if (!IsPowerOfTwo(fftLength))
        {
            throw new ArgumentException("FFT Length must be a power of two");
        }
        m = (int)Math.Log(fftLength, 2.0);
        this.fftLength = fftLength;
        fftBuffer = new Complex[fftLength];
        fftArgs = new FftEventArgs(fftBuffer);
        this.source = source;
    }

    static bool IsPowerOfTwo(int x)
    {
        return (x & (x - 1)) == 0;
    }


    public void Reset()
    {
        count = 0;
        maxValue = minValue = 0;
    }

    private void Add(float value)
    {
        if (PerformFFT && FftCalculated != null)
        {
            fftBuffer[fftPos].X = (float)(value * FastFourierTransform.HammingWindow(fftPos, fftLength));
            fftBuffer[fftPos].Y = 0;
            fftPos++;
            if (fftPos >= fftBuffer.Length)
            {
                fftPos = 0;
                // 1024 = 2^10
                FastFourierTransform.FFT(true, m, fftBuffer);
                FftCalculated(this, fftArgs);
            }
        }

        maxValue = Math.Max(maxValue, value);
        minValue = Math.Min(minValue, value);
        count++;
        if (count >= NotificationCount && NotificationCount > 0)
        {
            MaximumCalculated?.Invoke(this, new MaxSampleEventArgs(minValue, maxValue));
            Reset();
        }
    }

    public WaveFormat WaveFormat => source.WaveFormat;

    public int Read(float[] buffer, int offset, int count)
    {
        var samplesRead = source.Read(buffer, offset, count);

        for (int n = 0; n < samplesRead; n+=channels)
        {
            Add(buffer[n+offset]);
        }
        return samplesRead;
    }
}

public class MaxSampleEventArgs : EventArgs
{
    [DebuggerStepThrough]
    public MaxSampleEventArgs(float minValue, float maxValue)
    {
        MaxSample = maxValue;
        MinSample = minValue;
    }
    public float MaxSample { get; private set; }
    public float MinSample { get; private set; }
}

public class FftEventArgs : EventArgs
{
    [DebuggerStepThrough]
    public FftEventArgs(Complex[] result)
    {
        Result = result;
    }
    public Complex[] Result { get; private set; }
}

NAudio github存储库包含NAudioWpfDemo项目,该项目还包括频谱分析仪的实现。 我尝试在下面解释最重要的部分。 我在此答案中粘贴了相关代码,但是您需要看一下原始源代码才能完全理解它。

该演示项目使用WPF Polyline元素(请参见SpectrumAnalyser.xaml )来可视化FFT数据。

<UserControl x:Class="NAudioWpfDemo.SpectrumAnalyser">
    <Canvas Background="Black">
        <Polyline x:Name="polyline1" Stroke="Yellow" StrokeThickness="1"/>
    </Canvas>
</UserControl>

SpectrumAnalyser.xaml.cs中,您可以找到更新Polyline元素的代码。 方法Update(Complex[] fftResults)接收FFT数据,然后循环遍历FFT数据( fftResults数组)中的所有数据点。

for (int n = 0; n < fftResults.Length / 2; n+= binsPerPoint)
{
    // averaging out bins
    double yPos = 0;
    for (int b = 0; b < binsPerPoint; b++)
    {
        yPos += GetYPosLog(fftResults[n+b]);
    }
    AddResult(n / binsPerPoint, yPos / binsPerPoint);
}

...调用GetYPosLog(Complex c)以计算每个FFT数据点的dB值...

double intensityDB = 10 * Math.Log10(Math.Sqrt(c.X * c.X + c.Y * c.Y));

...并将转换后的数据点添加到方法AddResult(int index, double power)polyline1元素中

Point p = new Point(CalculateXPos(index), power);
polyline1.Points.Add(p);

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM