使用Java分离音频左右声道

1.音频属性相关

音频采样所得的PCM都含有三个要素:声道(channel)、采样率(sample rate)、采样位数、时长。

1.1.声道

记录声音时,如果每次生成一个声波数据,称为单声道;每次生成两个声波数据,称为双声道(立体声)。单声道的声音只能使用一个喇叭发声,双声道的PCM可以使两个喇叭同时发声(一般左右声道有分工),更能感受到空间效果。

1.2.采样率

单位时间内采集的样本数,即:采样周期的倒数,指两个采样之间的时间间隔。采样频率越高,声音质量越好,但同时占用的带宽越大。一般情况下,22KHz相当于普通FM的音质,44KHz相当于CD音质,目前的常用采样频率都不超过48KHz。

1.3.采样位数

表示一个样本的二进制位数,即:每个采样点用多少比特表示。计算机中音频的量化深度一般为4、8、16、32位(bit)等。例如:采样位数为8 bit时,每个采样点可以表示256个不同的采样值,而采样位数为16 bit时,每个采样点可以表示65536个不同的采样值。采样位数的大小影响声音的质量,采样位数越多,量化后的波形越接近原始波形,声音的质量越高,而需要的存储空间也越多;位数越少,声音的质量越低,需要的存储空间越少。一般情况下,CD音质的采样位数是16 bit,移动通信是8 bit。

1.4.帧

音频在量化得到二进制的码字后,需要进行变换,而变换(MDCT)是以块为单位(block)进行的,一个块由多个(120或128)样本组成。而一帧内会包含一个或者多个块。帧的常见大小有960、1024、2048、4096等。一帧记录了一个声音单元,它的长度是样本长度和声道数的乘积。FFmpeg中 AVFrame 结构体中的 nb_samples 代表的就是一帧中单个声道的音频样本数量。

数字音频文件大小(Byte) = 采样频率(Hz)× 采样时长(S)×(采样位数 / 8)× 声道数(单声道为1,立体声为2)

2.FFmpeg介绍

FFmpeg 是一个专业的多媒体框架,能够解码、编码、转码、复用、解复用、流式传输、过滤和播放几乎所有格式的媒体文件。

其核心就是 FFmpeg 程序本身,是一个基于命令行的视频和音频处理工具,多用于视频转码、基础编辑(修剪和合并)、视频缩放、后期效果制作等场景。

ffmpeg -i -hide_banner

2.1.FFmpeg 常用操作

`ffmpeg -i ``

1.转码

$ ffmpeg -i input.avi output.mp4

2.提取音频

$ ffmpeg -i input.mp4 output.mp3

3.调整分辨率

$ ffmpeg -i input.mp4 -s 1280x720 output.mp4

4.调整帧率

$ ffmpeg -i input.webm -c:a copy -c:v vp9 -r 30 output.mkv

5.提取音频

$ ffmpeg -i input.mp4 output.mp3

6.抓取图片

$ ffmpeg -i input.mp4 -r 1 -f image2 image-%2d.png

7.裁剪视频

$ ffmpeg -i input.mp4 -vf "crop=w:h:x:y" output.mp4

8.设置音频封面

$ ffmpeg -loop 1 -i inputimage.jpg -i inputaudio.wav -c:v libx264 -tune stillimage -c:a aac -b:a 192k -shortest output.mp4

9.截取视频片段

$ ffmpeg -i input.mp4 -ss 00:00:50 -codec copy -t 60 output.mp4

10.调整分辨率

$ ffmpeg -i input.mp4 -filter:v scale=1280:720 output.mp4

3.Jave2(Java音频视频编码器)

JAVE2是一个小的Java库,它将ffmpeg包装到java类中。 它是基于Carlo Pelliccia的杰作。 由于不再维护该代码,因此我们采用了该代码,并用当前版本替换了ffmpeg可执行文件,并修改了代码以使其与新的二进制文件一起使用。

在Java项目中可以通过maven或者gradle的方式引入依赖,这里使用maven讲解。

支持Windows,Mac,Linux等平台

ws.schild

jave-all-deps

2.7.3

4.JavaDemo

1.获取音频信息

import ws.schild.jave.EncoderException;

import ws.schild.jave.MultimediaInfo;

import ws.schild.jave.MultimediaObject;

import java.io.File;

/**

*

*

* @author leone

* @since 2020-08-08

**/

public class Test {

public static void main(String[] args) {

MultimediaInfo audioInfo = getAudioInfo(new File("/Users/leone/Downloads/mv.wav"));

assert audioInfo != null;

System.out.println("音频时长: " + audioInfo.getDuration());

System.out.println("音频声道数: " + audioInfo.getAudio().getChannels());

System.out.println("音频比特率: " + audioInfo.getAudio().getBitRate());

System.out.println("音频采样比特率: " + audioInfo.getAudio().getSamplingRate());

}

/**

* 获取音频详情

*

* @param audioFile

* @return

*/

public static MultimediaInfo getAudioInfo(File audioFile) {

MultimediaObject multimediaObject = new MultimediaObject(audioFile);

try {

return multimediaObject.getInfo();

} catch (EncoderException e) {

e.printStackTrace();

}

return null;

}

}

输出

15:19:12.619 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - Os name is isWindows: false isMac: true

15:19:12.624 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - Creating jave temp folder to place executables in

15:19:12.625 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - Executable path: /var/folders/31/qwvhtrys2gqdd9bwyb2dzwf80000gn/T/jave/ffmpeg-x86_64-2.7.3-osx

15:19:12.625 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - Need to copy executable to

15:19:12.625 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - Copy from resource to target

15:19:13.090 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - Target exists

15:19:13.179 [main] DEBUG ws.schild.jave.DefaultFFMPEGLocator - ffmpeg executable found: /var/folders/31/qwvhtrys2gqdd9bwyb2dzwf80000gn/T/jave/ffmpeg-x86_64-2.7.3-osx

15:19:13.182 [main] DEBUG ws.schild.jave.FFMPEGExecutor - About to execute /var/folders/31/qwvhtrys2gqdd9bwyb2dzwf80000gn/T/jave/ffmpeg-x86_64-2.7.3-osx -i /Users/leone/Downloads/mv.wav -hide_banner

15:19:14.267 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: Guessed Channel Layout for Input Stream #0.0 : stereo

15:19:14.268 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: Input #0, wav, from '/Users/leone/Downloads/mv.wav':

15:19:14.268 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: Metadata:

15:19:14.268 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: encoder : Lavf58.29.100

15:19:14.268 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: Duration: 00:03:35.64, bitrate: 1411 kb/s

15:19:14.268 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: Stream #0:0: Audio: pcm_s16le ([1][0][0][0] / 0x0001), 44100 Hz, stereo, s16, 1411 kb/s

15:19:14.270 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: At least one output file must be specified

15:19:14.270 [main] DEBUG ws.schild.jave.MultimediaObject - Output line: null

音频时长: 215640

音频声道数: 2

音频比特率: 1411000

音频采样比特率: 44100

2.音频格式转码

/**

* 音频文件转码

*

* @param source

* @return

*/

public File audioToWav(File source) {

try {

File target = File.createTempFile(UUID.randomUUID().toString(), ".tmp");

// Audio Attributes

AudioAttributes audio = new AudioAttributes();

// Encoding attributes

EncodingAttributes attrs = new EncodingAttributes();

attrs.setFormat("wav");

attrs.setAudioAttributes(audio);

// Encode

Encoder encoder = new Encoder();

MultimediaObject object = new MultimediaObject(source);

encoder.encode(object, target, attrs);

return target;

} catch (Exception ex) {

ex.printStackTrace();

}

return null;

}

5.使用Jave2API分离左右声道

/**

* @param originPath

* @param leftFilePath

* @param rightFilePath

* @return

*/

public static File[] doubleChannelSplit(String originPath, String leftFilePath, String rightFilePath) {

File leftFile = new File(leftFilePath);

File rightFile = new File(rightFilePath);

if (leftFile.exists()) {

leftFile.delete();

}

if (rightFile.exists()) {

rightFile.delete();

}

FFMPEGExecutor ffmpeg = locator.createExecutor();

ffmpeg.addArgument("-i");

ffmpeg.addArgument(originPath);

ffmpeg.addArgument("-map_channel");

ffmpeg.addArgument("0.0.0");

ffmpeg.addArgument(leftFilePath);

ffmpeg.addArgument("-map_channel");

ffmpeg.addArgument("0.0.1");

ffmpeg.addArgument(rightFilePath);

BufferedReader br = null;

try {

ffmpeg.execute();

br = new BufferedReader(new InputStreamReader(ffmpeg.getErrorStream()));

String line;

while ((line = br.readLine()) != null) {

logger.info(line);

}

return new File[]{leftFile, rightFile};

} catch (IOException e) {

e.printStackTrace();

} finally {

try {

if (br != null) {

br.close();

}

} catch (IOException e) {

e.printStackTrace();

}

}

return null;

}

6.最后

本来最开始是想用Java调用系统命令来完成音频声道分离的,但是考虑到兼容性,最后看了jave2的源码发现javaAPI中的操作也是基于调用jar包中的命令完成的,所以就采用这种方式实现。

java实现声道分离,使用Java分离音频左右声道相关推荐

  1. 使用Java分离音频左右声道

    使用Java分离音频左右声道 1.音频属性相关 音频采样所得的PCM都含有三个要素:声道(channel).采样率(sample rate).采样位数.时长. 1.1.声道 记录声音时,如果每次生成一 ...

  2. java短视频开发技术_看Java学员如何用前后端分离技术搭建短视频健身APP

    知了堂在教学过程中一直十分注重学员的实践能力,每阶段结束必有项目考核,全面检验分析学员学习效果.为更好的进行下一阶段的课程学习,本周一Java32班学员迎来了第二阶段的项目考核. 让我们来看看此次考核 ...

  3. Java中mysql的读写分离_mysql读写分离

    MySQL读写分离原理 MySQL的主从复制和MySQL的读写分离两者有着紧密联系,首先部署主从复制,只有复制完了,才能在此基础上进行数据的读写分离. 读写分离就是只在主服务器上写,只在从服务器上读, ...

  4. 计算机毕业设计-ssm+vue汽车销售管理系统-汽车商城(前后端分离)java代码

    计算机毕业设计-ssm+vue汽车销售管理系统-汽车商城(前后端分离)java代码 注意:该项目只展示部分功能,如需了解,评论区咨询即可. 1.开发环境 开发语言:Java 设计模式:MVC 架构:B ...

  5. 计算机毕业设计-基于ssm+vue的化妆品商城管理系统(前后端分离)java代码

    计算机毕业设计-基于ssm+vue的化妆品商城管理系统(前后端分离)java代码 注意:该项目只展示部分功能,如需了解,评论区咨询即可. 作者:IT跃迁谷 1.开发环境 开发语言:Java 设计模式: ...

  6. 在java中使用FFmpeg将图片和音频合成视频

    下面分享一个我之前在java中使用FFmpeg将图片和音频合成视频的demo. package com.xxx.console.videoProcess;import java.awt.image.B ...

  7. 如何在分割视频的基础上,分离视频中的音频

    一般我们在使用媒体梦工厂采集或者剪辑视频时,怎么在分割视频的基础上,分离视频中的音频呢?下面一起来试试. 预览视频效果 先预览一下用媒体梦工厂在批量分割视频的基础上,分割视频中的音频效果 一组视频一个 ...

  8. java 上传mp3文件大小,Java获取音频文件(MP3)的播放时长

    最近的一个项目需要按照时间播放mp3文件,例如,播放10分钟的不同音乐. 这就意味着我得事先知道mp3文件的播放时长,以决定播放几遍这个文件. 方案一:Java的方式 找第三方的库,真的感谢这些提供j ...

  9. java 微信小程序 语音识别成文字 音频格式转换 silk pcm wav

    最近有需求要把微信小程序里面的语音进行语音识别,然后搜搜,微信小程序的语音格式是silk 1.上传silk文件 2.下载silk-v3-decoder,通过名称把silk转换成讯飞可识别的wav文件 ...

最新文章

  1. 第十二周项目一-实现复数类中的运算符重载(3)
  2. “面试不败计划”:集合知识整体总结
  3. 最细的实现剖析:jQuery 2.0.3源码分析Deferred
  4. springer link:find the journals you need
  5. 使用CocoaPods出现 The `master` repo requires CocoaPods 0.32.1 - 问题解决
  6. java : enum、创建文件和文件夹、删除文件和文件夹、获得项目绝对路径、写入数据到excel中、java代码中两种路径符号写法、读取、写入text文件...
  7. python第三章上机实践_《机器学习Python实践》读书笔记-第三章
  8. 用Java编写约分最简公式,2013年Java方向C组第五题
  9. Linux /proc目录详解
  10. 看DLI服务4核心如何提升云服务自动化运维
  11. 定义一个Employee类,排序
  12. 多项式之和 Sn=a+aa+aaa+aaaa+.....+aaa.....aa(n个a)
  13. Matlab中的有限域计算
  14. Artifact XXX:war exploded: Artifact is being deployed, please wait...解决方法
  15. gitbook 入门教程之从零到壹发布自己的插件
  16. matlab 更换坐标轴_科学网—【Matlab】坐标轴的设置 - 叶瑞杰的博文
  17. 论文推荐:陈国生 实证化中医基础理论
  18. 猿创征文|MySQL入门到实战-基础篇
  19. vue脚手架和html,vue脚手架的作用是什么?
  20. tensorflow2 auto mpg汽车油耗预测实践(3.5节)

热门文章

  1. linux上p图工具,Linux下的图片转换工具ImageMagick
  2. 2022.11.5 J 课时1 小结
  3. (项目实战五)响应式案例和关于
  4. CNKI知网论文数据爬取
  5. idea没有jar包、maven没有dependencies、external libraries没有maven jar、libraries没有maven jar依赖
  6. 利用C语言编写程序计算数字测图中的三角高程实验
  7. 什么情况 进不去论坛
  8. Hive安装和使用 centos7
  9. Endnote20关联WPS(附上使用的b站链接)
  10. 自己动手写操作系统系列第1篇,从开机加电到切换保护模式