0

我在stackoverflow上搜索了最快的FFT算法,我发现了以下内容:

public class FFT {

      int n, m;

      // Lookup tables. Only need to recompute when size of FFT changes.
      double[] cos;
      double[] sin;

      public FFT(int n) {
          this.n = n;
          this.m = (int) (Math.log(n) / Math.log(2));

          // Make sure n is a power of 2
          if (n != (1 << m))
              throw new RuntimeException("FFT length must be power of 2");

          // precompute tables
          cos = new double[n / 2];
          sin = new double[n / 2];

          for (int i = 0; i < n / 2; i++) {
              cos[i] = Math.cos(-2 * Math.PI * i / n);
              sin[i] = Math.sin(-2 * Math.PI * i / n);
          }

      }

      public void fft(double[] x, double[] y) {
          int i, j, k, n1, n2, a;
          double c, s, t1, t2;

          // Bit-reverse
          j = 0;
          n2 = n / 2;
          for (i = 1; i < n - 1; i++) {
              n1 = n2;
              while (j >= n1) {
                  j = j - n1;
                  n1 = n1 / 2;
              }
              j = j + n1;

              if (i < j) {
                  t1 = x[i];
                  x[i] = x[j];
                  x[j] = t1;
                  t1 = y[i];
                  y[i] = y[j];
                  y[j] = t1;
              }
          }

          // FFT
          n1 = 0;
          n2 = 1;

          for (i = 0; i < m; i++) {
              n1 = n2;
              n2 = n2 + n2;
              a = 0;

              for (j = 0; j < n1; j++) {
                  c = cos[a];
                  s = sin[a];
                  a += 1 << (m - i - 1);

                  for (k = j; k < n; k = k + n2) {
                      t1 = c * x[k + n1] - s * y[k + n1];
                      t2 = s * x[k + n1] + c * y[k + n1];
                      x[k + n1] = x[k] - t1;
                      y[k + n1] = y[k] - t2;
                      x[k] = x[k] + t1;
                      y[k] = y[k] + t2;
                  }
              }
          }
      }
    }

我的问题是我有一个 MediaRecorder 对象来捕获音频,如下所示:

if (mRecorder == null) {
            mRecorder = new MediaRecorder();
            mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
            mRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
            mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);

            mRecorder.setOutputFile("/dev/null");

            try {
                mRecorder.prepare();
            } catch (IllegalStateException e) {
                Log.e("error", "IllegalStateException");
            } catch (IOException e) {
                Log.e("error", "IOException");
                ;
            }

            mRecorder.start();
        }

现在我想在我捕获的音频上使用这个 FFT 算法,并在均衡器或其他东西上显示结果。我怎么能这样做?

4

1 回答 1

2

MediaRecorder 不会让您直接访问音频缓冲区,但如果您使用 AudioRecord,则可以。如果您必须使用 MediaRecorder,则可以将其保存到文件中,然后再次重新读取该文件。

有人在这里研究过一个例子,在 Android 中捕获声音以进行分析和可视化频率

于 2013-09-26T10:14:51.307 回答