0

0

Java怎么在播放时显示音频频谱 Java实时频谱分析仪实现教程【代码】

畫卷琴夢

畫卷琴夢

发布时间:2026-01-19 14:59:02

|

919人浏览过

|

来源于php中文网

原创

纯Java实现实时音频频谱需手动FFT和可视化,易卡顿延迟高;TarsosDSP最省事,支持自动分帧加窗FFT及回调输出,但须显式设采样率、50%重叠、汉宁窗补偿、dB转换与EDT线程同步。

java怎么在播放时显示音频频谱 java实时频谱分析仪实现教程【代码】

Java 本身没有内置的实时音频频谱绘制能力,javax.sound.sampled 只能采集原始 PCM 数据,频谱计算和可视化必须手动实现——这意味着你得自己做 FFT(快速傅里叶变换),再把结果映射到图形上。不借助第三方音频处理库(如 TarsosDSP)或 JNI 封装(如 PortAudio + JNA),纯 Java 实现容易卡顿、延迟高、频谱不准。

用 TarsosDSP 做实时音频频谱最省事

TarsosDSP 是纯 Java 的音频处理库,自带 AudioDispatcherFFTSpectrumAnalyzer,适合桌面端轻量级实时分析。它从麦克风读取数据后自动分帧、加窗、FFT,并通过回调输出频谱幅值数组。

  • 必须用 AudioFormat 显式指定采样率(推荐 4410048000),否则 AudioSystem.getTargetDataLine() 可能返回不支持的格式导致静音
  • Overlap 设为 50%(即 bufferSize / 2)能提升时域分辨率,避免频谱跳变
  • 频谱点数 = bufferSize / 2 + 1(实数 FFT 输出),不是 bufferSize 全长
  • 绘图建议用 SwingpaintComponent 配合双缓冲,别在事件线程里直接 repaint()

FFT 输入前必须加汉宁窗(Hanning window)

原始音频帧直接 FFT 会产生频谱泄漏,高频能量“拖尾”,峰位偏移。TarsosDSP 默认不加窗,需手动包装 FloatBuffer 数据。

  • 窗口函数用 float[i] *= 0.5 - 0.5 * Math.cos(2 * Math.PI * i / (length - 1))
  • 加窗后要对幅值做补偿(通常 ×2),否则低频衰减明显
  • 避免用矩形窗(即不加窗)——哪怕只是调试,也会看到底噪抬升、谐波分裂

Swing 绘制频谱条时注意坐标和缩放

频谱 Y 轴是幅值(非分贝),但人耳对数响应,直接画线性值会看不到低能量频段。必须转 dB = 20 * log10(|X[i]| + 1e-9),再归一化到控件高度。

讯飞智作-虚拟主播
讯飞智作-虚拟主播

讯飞智作是一款集AI配音、虚拟人视频生成、PPT生成视频、虚拟人定制等多功能的AI音视频生产平台。已广泛应用于媒体、教育、短视频等领域。

下载

立即学习Java免费学习笔记(深入)”;

  • 不要用 Math.log10(0)——未加保护会得 -Infinity,绘图线程崩溃
  • X 轴频率分布非线性:索引 i 对应频率是 i * sampleRate / bufferSize;想看 20Hz–20kHz,bufferSize 至少取 2048(44.1kHz 下最低分辨约 21.5Hz)
  • 每帧重绘前清空 Graphics2D 背景,否则残留拖影;用 setComposite(AlphaComposite.Clear) 清屏比 fillRect 更稳
import be.tarsos.dsp.AudioDispatcher;
import be.tarsos.dsp.AudioEvent;
import be.tarsos.dsp.io.jvm.JVMAudioInputStream;
import be.tarsos.dsp.pitch.PitchDetectionHandler;
import be.tarsos.dsp.pitch.PitchDetectionResult;
import be.tarsos.dsp.pitch.PitchProcessor;

import javax.swing.; import java.awt.; import java.awt.geom.Rectangle2D;

public class SpectrumPanel extends JPanel implements PitchDetectionHandler { private final int width = 800; private final int height = 300; private final double[] spectrum = new double[1024]; private final float[] buffer = new float[2048];

public SpectrumPanel() {
    setPreferredSize(new Dimension(width, height));
    setBackground(Color.BLACK);
}

@Override
protected void paintComponent(Graphics g) {
    super.paintComponent(g);
    Graphics2D g2 = (Graphics2D) g;
    g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);

    for (int i = 0; i < spectrum.length && i < width; i++) {
        double y = height - (spectrum[i] * height * 0.8);
        g2.setColor(new Color(0, (int)(spectrum[i]*200), 255));
        g2.fill(new Rectangle2D.Double(i, y, 1, height - y));
    }
}

@Override
public void handlePitch(PitchDetectionResult result, AudioEvent e) {
    float[] audioBytes = e.getFloatBuffer();
    System.arraycopy(audioBytes, 0, buffer, 0, Math.min(audioBytes.length, buffer.length));

    // Apply Hanning window
    for (int i = 0; i < buffer.length; i++) {
        buffer[i] *= 0.5 - 0.5 * Math.cos(2 * Math.PI * i / (buffer.length - 1));
    }

    // Simple magnitude spectrum (real FFT assumed)
    for (int i = 0; i < Math.min(spectrum.length, buffer.length/2+1); i++) {
        double re = 0, im = 0;
        // Dummy FFT — in real use: feed to FFT class or use Tarsos' FFT
        // This is placeholder logic; actual impl needs proper FFT
        spectrum[i] = Math.max(0.01, Math.sqrt(re*re + im*im) * 2);
        spectrum[i] = 20 * Math.log10(spectrum[i] + 1e-9); // to dB
        spectrum[i] = Math.min(1.0, Math.max(0.0, (spectrum[i] + 60) / 60)); // normalize 0–1
    }
    repaint();
}

public static void main(String[] args) {
    JFrame frame = new JFrame("Spectrum Analyzer");
    SpectrumPanel panel = new SpectrumPanel();
    frame.add(panel);

    AudioDispatcher dispatcher = AudioDispatcher.fromDefaultMicrophone(2048, 1024);
    dispatcher.addAudioProcessor(new PitchProcessor(PitchProcessor.PitchEstimationAlgorithm.FFT_YIN, 44100, 2048, panel));
    new Thread(dispatcher).start();

    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.pack();
    frame.setVisible(true);
}

}

真正卡住的地方不在 FFT 算法本身,而在音频流与 UI 线程的同步:Swing 不是线程安全的,repaint() 必须在 EDT 中触发,但音频回调在后台线程。上面示例用了 TarsosDSP 的 PitchProcessor 包装器来桥接,实际项目中更稳妥的做法是用 SwingUtilities.invokeLater() 包裹 repaint(),否则偶尔会抛 NullPointerException 或界面冻结。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

837

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

741

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

736

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

8

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.7万人学习

C# 教程
C# 教程

共94课时 | 7万人学习

Java 教程
Java 教程

共578课时 | 47.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号