• Android MediaRecorder录音录像 暂停 继续录音 播放 ARM格式(音频一)

    https://blog.csdn.net/WHB20081815/article/details/88778605

    Android 录音机小米商业项目开源代码 AudioRecord录音暂停 播放 Wav格式(音频二)

    https://blog.csdn.net/WHB20081815/article/details/88778623

    Android 录音机小米商业项目开源代码 pcm转AAC硬编码 录音暂停 播放 (音频三)

    https://blog.csdn.net/WHB20081815/article/details/88778634

    Android 录音机商业项目开源代码 pcm转AAC软编码 录音暂停 播放 (音频四)

    https://blog.csdn.net/WHB20081815/article/details/88778641

    Android 高仿唱吧 咔拉ok 商业项目开源代码 K歌合成 伴奏录音合成MP3(音频五)

    https://blog.csdn.net/WHB20081815/article/details/88778652

  • 音频混合:一段音频和另一段音频存在相同的区间,两者会有同时播放的区间。

情景一

假设A音频40秒,B音频20秒,B音频数据拼接到A音频后面,得到60秒的C音频文件。

这种情况最简单了,新建音频文件C,将A音频的PCM数据复制到C音频文件上,再将B音频的PCM数据复制到C音频文件上,然后为C音频写上wav文件头信息,得到可播放的WAV文件。

情景二

假设A音频40秒,B音频20秒,B音频数据插入到A音频10秒的地方,得到60秒的C音频文件。

这种情况稍微复杂点,新建音频文件C,将A音频前10秒的PCM数据复制到C音频文件上,再将B音频的PCM数据复制到C音频文件上,再将A音频后30秒的PCM数据复制到C音频文件上,最后为C音频写上wav文件头信息,得到可播放的WAV文件。

情景三

假设A音频40秒,B音频20秒,B音频5至15秒的数据插入到A音频10秒的地方,得到50秒的C音频文件。

这种情况更复杂,也是最常见的插入场景,裁剪B音频并插入到A音频的某个位置,这里涉及到B音频数据的裁剪,当然原理其实也是简单的,计算出B音频5秒和10秒对应的文件数据位置,然后复制这个区间的数据到C上,针对A文件的数据,也是同样道理。

情景四

A音频和B音频中多段数据相互拼接

这种情况,原理同上面一样,只要知道指定时间对应的数据是什么,就可以实现自由拼接了。

音频拼接的实现参考我的Github项目 AudioEdit,这里我就不贴具体代码了。

音频混合

音频混合是指一段音频和另一段音频合在一起,能够同时播放,比如最常见的人声录音和背景音乐的合成,可以得到一首人声歌曲。
音频混合的原理是

音频混合原理: 量化的语音信号的叠加等价于空气中声波的叠加。

也就是说将输入的每段音频的某个时间点的采样点数值进行相加,即可将声音信号加入到输出的音频中。

音频采样点数值的大小是(-32768,32767),对应short的最小值和最大值,音频采样点数据就是由一个个数值组成的的。如果单纯叠加,可能会造成相加后的值会大于32767,超出short的表示范围,也就是溢出,所以在音频混合上回采用一些算法进行处理。下面列举下简单的混合方式。

直接叠加法

A(A1,A2,A3,A4)和B(B1,B2,B3,B4)叠加后求平均值,得到C((A1+B1),(A2+B2),(A3+B3),(A4+B4))
这种情况,输出的音频中A和B音频数据都可以以相同声音大小播放,但是可能出现溢出的情况。假设A音频指定时间点的某段采样数据是(23,67,511,139,307),B音频对应该时间点的采样数据是(1101,300,47,600,22),那么两者直接叠加的话,得到的采样数据是(1124,367,558,739,329),这个短采样数据就是两者声音混合的数据了。

叠加后求平均值

A(A1,A2,A3,A4)和B(B1,B2,B3,B4)叠加后求平均值,得到C((A1+B1)/2,(A2+B2)/2,(A3+B3)/2,(A4+B4)/2)
这样可以避免出现溢出的情况,但是会出现两者声音会比之前单独的声音小了一半,比如人声和背景音乐混合,导致输出的音频中,人声小了一半,背景音乐也小了一半,这种情况可能就不是想要的效果,特别是多段音频混合的情况。

权值叠加法

A(A1,A2,A3,A4)和B(B1,B2,B3,B4)权值叠加,A权值为x,B权值为y,得到C((A1 * x+B1 * y),(A2 * x+B2 * y),(A3 * x+B3 * y),(A4 * x+B4 * y))
这样可以更方便条件A和B的音量的大小,比如A的权值为1.2,B的权值为0.8,那么A的声音相对提高了,B的声音相对减弱了。严格来说,直接叠加法和叠加求平均值法都属于该类型。

此外还有各种更复杂的混合算法,如动态权值法,A和B的权值会根据当前时刻采样点数值的大小进行动态变化,得到一个动态增益和衰减的混合方式。

下面是直接叠加法的实现,需要注意short值要按大端存储的方式计算,存储时按大端方式存储。

  /*** 叠加合成器* @author Darcy*/private static class AddAudioMixer extends MultiAudioMixer{@Overridepublic byte[] mixRawAudioBytes(byte[][] bMulRoadAudioes) {if (bMulRoadAudioes == null || bMulRoadAudioes.length == 0)return null;byte[] realMixAudio = bMulRoadAudioes[0];if(bMulRoadAudioes.length == 1)return realMixAudio;for(int rw = 0 ; rw < bMulRoadAudioes.length ; ++rw){if(bMulRoadAudioes[rw].length != realMixAudio.length){Log.e("app", "column of the road of audio + " + rw +" is diffrent.");return null;}}//row 代表参与合成的音频数量//column 代表一段音频的采样点数,这里所有参与合成的音频的采样点数都是相同的int row = bMulRoadAudioes.length;int coloum = realMixAudio.length / 2;short[][] sMulRoadAudioes = new short[row][coloum];//PCM音频16位的存储是大端存储方式,即低位在前,高位在后,例如(X1Y1, X2Y2, X3Y3)数据,它代表的采样点数值就是((Y1 * 256 + X1), (Y2 * 256 + X2), (Y3 * 256 + X3))for (int r = 0; r < row; ++r) {for (int c = 0; c < coloum; ++c) {sMulRoadAudioes[r][c] = (short) ((bMulRoadAudioes[r][c * 2] & 0xff) | (bMulRoadAudioes[r][c * 2 + 1] & 0xff) << 8);}}short[] sMixAudio = new short[coloum];int mixVal;int sr = 0;for (int sc = 0; sc < coloum; ++sc) {mixVal = 0;sr = 0;//这里采取累加法for (; sr < row; ++sr) {mixVal += sMulRoadAudioes[sr][sc];}//最终值不能大于short最大值,因此可能出现溢出sMixAudio[sc] = (short) (mixVal);}//short值转为大端存储的双字节序列for (sr = 0; sr < coloum; ++sr) {realMixAudio[sr * 2] = (byte) (sMixAudio[sr] & 0x00FF);realMixAudio[sr * 2 + 1] = (byte) ((sMixAudio[sr] & 0xFF00) >> 8);}return realMixAudio;}
  1. 需要确保A音频和B音频的采样位数一致。例如A音频是16位采样位数,B音频是8位采样位数,那么这时是不能直接拼接的,需要转换成相同的采样位数,才能做后续操作。
  2. 需要确保A音频和B音频的采样率一致。这个在录音和歌曲拼接时要特别注意,假如录音的音频频率是16000,歌曲的音频是44100,那么两者也是不能直接拼接的,需要转换成相同的采样率,转换采样率可以使用resample库。
  3. 需要确保A音频和B音频的声道数一致。当然这个并不是指单声道和双声道的音频不能合成了,事实上录音音频通常是单声道的,而歌曲通常是双声道的。单声道和双声道音频合成,一般是按双声道为基准,需要将单声道音频转换成双声道音频,转换原理也简单,将单声道的采样点数据多复制一份,比如将单声道的ABCD数据转换成双声道的AABBCCDD数据。

那么我们可能会有疑问,如果A音频和B音频的采样率位数,采样率,声道数不一样的话,合成后是有效的音频文件吗?这个其实是有效的,同样可以播放,但是会造成合成后的音频不同部分的音频播放速度不一样,例如单声道的A和双声道的B拼接,会造成A部分的播放速度比B的播放速度快一倍,而B的播放速度是正常的。

原理:

    网上查了好多资料,看了某大神的例子,选出了这个最简单的方式,amr格式的录音文件的合并,只需要把每个录音文件的二进制数据的前6位数据给删除,然后一起保存起来就实现了无缝拼接。比如录了两个文件,a.amr和b.amr,只需要把b.amr的头6位数据删除,然后把两个文件的字节流合并成一个文件,就实现了无缝拼接。

录音完成后,调用开源工具(Mad)实现PCM合成输出到MP3文件.

主要调用的合成方法:

}

注意事项

音频的拼接和混音,有一些是需要注意和处理的。

/***

* 方法描述:本地方法调用JNI合并mp3PCM与sourcePCM

* @param sourcePCM

* @param mp3PCM

* @param mixPCM

* @return

*/

public static native int mix2PCMToPCM(String sourcePCM, String mp3PCM, String mixPCM);

String recordPCMPath = SongUtil.getRecordSingPCMPath(songId); //录音生成的PCM文件

String accompanyPCMPath = SongUtil.getAccompanySongPCMPath(songId); //伴奏解码生成的PCM文件

String mixPCMPath = SongUtil.getMixSingPCMPath(songId); //合成后的PCM文件

String mixMP3Path = SongUtil.getMixSingMp3Path(songId); //合成后的MP3文件

// 混音

int code = SongEncodeUtil.mix2PCMToPCM(recordPCMPath, accompanyPCMPath, mixPCMPath);

if (code == 0) {

// 转换混合后音频格式 TO mp3

int i = SimpleLame.convert(mixPCMPath, mixMP3Path, m_in_buf_size);

Log.i(SingSingleActivity.this.getClass().getName(), "转换" + i + "混音完成");

saveMp3File(mixMP3Path);

---------------------

https://www.jianshu.com/p/6c05968d256b

demo下载地址:

https://download.csdn.net/download/h3c4lenovo/4684013

Android 高仿唱吧 咔拉ok 商业项目开源代码 K歌合成 伴奏录音合成MP3(音频五)相关推荐

  1. Android 录音机小米商业项目开源代码 AudioRecord录音暂停 播放 Wav格式(音频二)

    Android MediaRecorder录音录像 暂停 继续录音 播放 ARM格式(音频一) https://blog.csdn.net/WHB20081815/article/details/88 ...

  2. android高仿网易新闻上下拉刷新,仿网易新闻最新版的下拉刷新

    今天一天做了一个仿照网易的下拉刷新,先上效果图: 代码还有一点问题.我先说怎么用把: 1.   布局文件. xmlns:tools="http://schemas.android.com/t ...

  3. (android高仿系列)今日头条 --新闻阅读器 (一)

    在模仿中循序渐进,以程序员角度去看待每一个APP是如何实现的,它有什么优缺点,并从中提升自己. 之前发现很多人在群里面.论坛上求网易新闻客户端的源码,之后我就去下了个网易新闻客户端和今日头条新闻客户端 ...

  4. Android高仿马蜂窝Tabbar波浪线

    文章目录 简介 效果 思路 关键代码 先画完整的弧线,图中红色曲线 再画指示器部分的弧线,图中深蓝色弧线 不足 源码[github](https://github.com/Super-Bin/Horn ...

  5. android高仿全民直播、多窗口管理、图标效果、开发框架,下拉上滑、UI框架等源码...

    Android精选源码 android仿UC浏览器多窗口管理界面 android高仿全民直播项目源码 android微信小程序设计模式,多任务窗体运行源码 Android项目源码在线课堂教育UI框架源 ...

  6. android高仿全民直播、多窗口管理、图标效果、开发框架,下拉上滑、UI框架等源码

    Android精选源码 android仿UC浏览器多窗口管理界面 android高仿全民直播项目源码 android微信小程序设计模式,多任务窗体运行源码 Android项目源码在线课堂教育UI框架源 ...

  7. (android高仿系列)今日头条 --新闻阅读器 (二)

    高仿今日头条 --- 第一篇:(android高仿系列)今日头条 --新闻阅读器 (一) 上次,已经完成了头部新闻分类栏目的拖动效果. 这篇文章是继续去完善APP 今日头条  这个新闻阅读器的其他功能 ...

  8. android+高仿视频录制,android高仿微信视频编辑页

    android高仿微信视频编辑页-视频多张图片提取 上一篇中介绍了有关视频提取图片的知识点,如果对这个不太了解 建议看下android提取视频多张图片和视频信息之前这篇. 这里实现的是仿微信的视频编辑 ...

  9. android qq红点,Android高仿QQ小红点功能

    先给大家展示下效果图: 绘制贝塞尔曲线: 主要是当在一定范围内拖拽时算出固定圆和拖拽圆的外切直线以及对应的切点,就可以通过path.quadTo()来绘制二阶贝塞尔曲线了~ 整体思路: 1.当小红点静 ...

最新文章

  1. H1标签对网站优化有什么作用?
  2. GDIDrawing3——GDI+绘图(三)
  3. MacOS 系统使用命令安装软件包
  4. linux C语言 文件相关知识01
  5. JavaEE编码规范
  6. 一对电话线传输100M带宽不再是问题
  7. php限制上传类型,php 上传类型限制的简单示例
  8. android之sqlite增、删、改,查
  9. dynamips虚拟服务:找不到指定设备
  10. 手机在我状态查询易语言代码
  11. 风尚云网学习-js实现禁用右键以及F12
  12. MySQL局域网连接失败问题解决
  13. CF Make Cents?
  14. 阿里云建站百度收录吗?
  15. Android吃鸡 3dtouch,绝地求生刺激战场3Dtouch怎么用 3Dtouch安卓手机可以用吗
  16. Chrome打开摄像头权限
  17. LVGL系列(四)概述 之 位置、尺寸和布局
  18. 疯癫的我到了巅峰 -- 我的成人礼
  19. Homestead 安装 phpMyAdmin 作为数据库管理客户端 — Laravel 实战 iBrand API 教程
  20. Unity更改模型中心和旋转中心

热门文章

  1. 二叉树,满二叉树,完全二叉树 概念及其性质
  2. python中继承是什么意思,解释一下python中的继承
  3. vue.runtime.esm.js:1737 TypeError: _self.$scopedSlots.default is not a function
  4. NS3 sixth.cc注释
  5. 超详细:前端 2019 学习路线(B站视频)
  6. 数据库——ltrim()和rtrim()函数
  7. I2C通信协议:了解I2C Primer、PMBus和SMBus
  8. 清理Xcode DerivedData文件
  9. 程序员怎么跟领导提涨薪?
  10. 白帽子黑客与网络安全工程师带你:远程桌面漏洞利用与Windows7计算机攻防实战