0

0

Java如何从麦克风录音 Java Sound API实现实时录音功能【代码实例】

月夜之吻

月夜之吻

发布时间:2026-01-25 08:03:08

|

614人浏览过

|

来源于php中文网

原创

AudioSystem.getAudioInputStream抛UnsupportedAudioFileException主因是误将麦克风TargetDataLine等原始PCM流当作文件传入,因其无WAV/AIFF等文件头;正确做法是直接调用line.read()读取字节。

java如何从麦克风录音 java sound api实现实时录音功能【代码实例】

AudioSystem.getAudioInputStream 为什么总抛 UnsupportedAudioFileException

常见原因是传入了不支持的音频格式或设备流。Java Sound API 的 AudioSystem.getAudioInputStream 只能处理已知编码的文件(如 WAV、AIFF),**不能直接用于麦克风实时流**。试图对 TargetDataLine 调用它会失败——因为麦克风数据是原始 PCM 流,没有文件头。

正确做法是跳过这一步,直接从 TargetDataLine 读取字节

AudioFormat format = new AudioFormat(
    AudioFormat.Encoding.PCM_SIGNED,
    44100.0f, // sample rate
    16,       // sample size in bits
    2,        // channels
    4,        // frame size
    44100.0f, // frame rate
    false     // big endian
);
DataLine.Info info = new DataLine.Info(TargetDataLine.class, format);
TargetDataLine line = (TargetDataLine) AudioSystem.getLine(info);
line.open(format);
line.start(); // 开始采集

byte[] buffer = new byte[4096]; int bytesRead; while (recording) { bytesRead = line.read(buffer, 0, buffer.length); // 直接读,别 wrap 成 InputStream // 处理 buffer 中的 PCM 数据 }

如何避免 LineUnavailableException 启动失败

这个异常几乎总是因为:同一时间有其他程序占用了麦克风(如 Zoom、Teams、系统录音机),或 JVM 没有权限访问音频设备(尤其在 macOS 或 Linux 的 sandbox 环境中)。

  • 检查是否已有 TargetDataLine 在运行:调用前先 line.isRunning() 或确保 close() 旧实例
  • 显式指定设备而非依赖默认:用 AudioSystem.getMixer(null) 列出所有混音器,挑出带 "Microphone""Capture" 字样的
  • Linux 下可能需加入 audio 用户组;macOS 需在「系统设置 → 隐私与安全性 → 麦克风」中允许 Java 应用

实时录音时 CPU 占用飙升,怎么压低延迟又不卡顿

核心矛盾在于缓冲区大小:buffer 太小 → 频繁 read() 调用,上下文切换多;太大 → 延迟高、响应滞后。4096 字节(约 23ms @ 44.1kHz/16bit/stereo)是较稳妥起点。

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

更关键的是别在 read() 循环里做重操作:

腾讯AI 开放平台
腾讯AI 开放平台

腾讯AI开放平台

下载
  • 不要在循环内新建对象(如 new ByteArrayOutputStream()
  • 避免同步 IO(如边录边写磁盘 WAV 文件)——改用双缓冲队列 + 独立写线程
  • 若需实时 FFT 或降噪,优先用 float[] 批量转换(用 AudioFormatgetSampleSizeInBits()isBigEndian() 正确解析 PCM)

保存为 WAV 文件必须手动写 header

Java Sound API 不提供“把 raw PCM 流自动封装成 WAV”的工具类。WAV 是 RIFF 容器格式,header 固定 44 字节,含采样率、位深、声道数等字段。漏写或写错会导致播放器无法识别。

最简 header 构造逻辑(适用于 PCM_SIGNED, 16-bit, stereo):

public static byte[] wavHeader(int byteRate, int audioDataLength) {
    byte[] header = new byte[44];
    // "RIFF"
    header[0] = 'R'; header[1] = 'I'; header[2] = 'F'; header[3] = 'F';
    // file size = 36 + audioDataLength
    writeInt(header, 4, 36 + audioDataLength);
    // "WAVE"
    header[8] = 'W'; header[9] = 'A'; header[10] = 'V'; header[11] = 'E';
    // "fmt "
    header[12] = 'f'; header[13] = 'm'; header[14] = 't'; header[15] = ' ';
    // format chunk size = 16
    writeInt(header, 16, 16);
    // format = 1 (PCM)
    writeShort(header, 20, (short) 1);
    // channels = 2
    writeShort(header, 22, (short) 2);
    // sampleRate = 44100
    writeInt(header, 24, byteRate / 4); // 注意:byteRate = sampleRate * channels * bitsPerSample/8
    // byteRate
    writeInt(header, 28, byteRate);
    // blockAlign = channels * bitsPerSample/8
    writeShort(header, 32, (short) 4);
    // bitsPerSample = 16
    writeShort(header, 34, (short) 16);
    // "data"
    header[36] = 'd'; header[37] = 'a'; header[38] = 't'; header[39] = 'a';
    // subchunk2Size = audioDataLength
    writeInt(header, 40, audioDataLength);
    return header;
}

header 写完后,再追加原始 PCM byte[] ——顺序不能反,长度不能错。

真实项目里最容易被忽略的,是不同平台对 TargetDataLine 缓冲行为的差异:Windows 上默认 buffer 可能偏大,Linux ALSA 有时会静默丢帧,而 macOS CoreAudio 对 open()millisecondTimeout 参数更敏感。别假设一次参数适配全平台。

相关专题

更多
java
java

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

845

2023.06.15

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

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

745

2023.07.05

java自学难吗
java自学难吗

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

740

2023.07.31

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

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

397

2023.08.01

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

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

420

2023.08.02

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

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

447

2023.08.02

java有什么用
java有什么用

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

431

2023.08.02

java在线网站
java在线网站

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

16947

2023.08.03

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

25

2026.01.23

热门下载

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

精品课程

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

共48课时 | 7.7万人学习

Git 教程
Git 教程

共21课时 | 3万人学习

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

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