Java调用JAVE2分离音频需先确保FFmpeg路径正确,构造Encoder时传入MultimediaObject和AudioAttributes,并设audio.setCodec("copy")、bitRate/channels为0、video.setCodec(null),显式指定format如"mp3",否则默认重编码;多音轨或精细控制应直接调用ffmpeg。

Java调用JAVE2分离视频中的音频流要先确认FFmpeg可执行文件路径
JAVE2本身不自带FFmpeg,它只是Java封装层,底层必须依赖系统已安装的FFmpeg二进制文件。如果ffmpeg命令在终端/命令提示符中无法直接运行,JAVE2会抛出java.io.IOException: Cannot run program "ffmpeg": error=2, No such file or directory。
- Linux/macOS:确保
ffmpeg在$PATH中,或显式传入绝对路径(如/usr/local/bin/ffmpeg) - Windows:推荐将
ffmpeg.exe放在项目根目录或resources/下,并用getClass().getResource("/ffmpeg.exe").getPath()获取路径(注意URL转义问题,可能需用URLDecoder.decode(..., "UTF-8")) - JAVE2构造
Encoder时必须传入new MultimediaObject(new File(videoPath))和new AudioAttributes(),否则默认行为是转码而非提取
使用AudioAttributes指定纯音频提取而非转码
很多人误以为只要设了AudioAttributes就自动“提取”,其实JAVE2默认仍会重编码——除非显式禁用视频并设音频编码为copy。否则即使输入是MP3,输出也可能是AAC+重采样,白白损失质量且变慢。
- 必须调用
audioAttrs.setCodec("copy"),这是无损提取的关键 - 必须调用
audioAttrs.setBitRate(0)和audioAttrs.setChannels(0),避免JAVE2强行插入默认值触发重编码 - 视频属性必须设为
null或显式禁用:videoAttrs.setCodec(null),否则JAVE2可能尝试解码视频帧造成失败 - 输出格式由目标文件扩展名决定(如
output.mp3),但某些容器(如.m4a)需要对应format("mp4"),否则报Invalid argument
常见错误:Extracting audio fails with “Unable to find a suitable output format”
这个错误几乎总是因为输出路径扩展名与JAVE2推断的format不匹配。例如写output.aac但没指定format,JAVE2可能按flv处理;或者写output.wav却用了setCodec("copy")(WAV不支持流复制,必须重编码)。
- 安全做法:显式调用
encodingAttributes.setFormat("mp3")(对应.mp3)、"aac"(对应.aac或.m4a)、"wav"(此时必须去掉setCodec("copy")) - 若输入视频含多音轨,JAVE2默认只取第0轨;需提取特定音轨得改用FFmpeg原生命令(JAVE2不暴露
-map 0:a:1等参数) - Windows下路径含中文或空格时,务必用
File.getCanonicalPath()规范化,否则ProcessBuilder启动失败
AudioAttributes audio = new AudioAttributes();
audio.setCodec("copy");
audio.setBitRate(0);
audio.setChannels(0);
VideoAttributes video = new VideoAttributes();
video.setCodec(null); // 关键:禁用视频处理
EncodingAttributes attrs = new EncodingAttributes();
attrs.setFormat("mp3");
attrs.setAudioAttributes(audio);
attrs.setVideoAttributes(video);
Encoder encoder = new Encoder();
encoder.encode(
new MultimediaObject(new File("input.mp4")),
new File("output.mp3"),
attrs
);
JAVE2对FFmpeg参数控制较弱,真正需要精确控制(如提取某条音轨、压制特定采样率、跳过前5秒)时,直接拼接ProcessBuilder调用ffmpeg更可靠——毕竟JAVE2只是薄封装,不是抽象层。










