AndroidCoding-关于录音暂停、重新录音(amr merge)、放大录入音量同时保证文件压缩率(wav2amr)

之前公司有个录音的需求,需要用户在使用某个功能的时候实时录用。在App退到后台时候暂停录音,回到前台时候重新录音。
一开始的的做法是使用MediaRecorder去录音,app挂到后台时候停止录音,回到前台重新录音。在录音动作结束的时候,将多个amr文件合并成一个单独的amr文件。下面是amr文件合并的工具类

/**
 * author: wangzh
 * create: 2018/11/26 14:47
 * description: amr录音文件合并工具,因为RAW_AMR,AMR_NB的头文件长度固定为6个字节,AMR_WB的头文件长度为9个字节。
 * 所以使用RandomAccessFile就可以轻松的完成amr文件的合并了。
 * version: 1.0
 */
public class AMRMergeUtil {


    public static final int SEEK_LENGTH_AMR_NB = 6;
    public static final int SEEK_LENGTH_AMR_WB = 9;

    /**
     * 缓冲的数组大小
     */
    private static final int BUFF_BYTE_ARRAY_LENGTH = 4096;

    /**
     * @param rootDir    存放着需要合并的amr录音文件的父目录绝对路径
     * @param seekLength 跳过的头部的长度,RAW_AMR,AMR_NB的头文件长度固定为6个字节,AMR_WB的头文件长度为9个字节。
     * @return 返回{@link #Result}对象
     */
    public static Result merge(String rootDir, int seekLength) throws IOException {
        Result result = new Result();
        if (TextUtils.isEmpty(rootDir)) {
            result.setResult(false);
            result.setMessage("传入的文件夹路径为空");
            return result;
        }
        File sourceDir = new File(rootDir);
        if (!sourceDir.exists() || sourceDir.isFile()) {
            result.setResult(false);
            result.setMessage("传入的文件夹不存在或者不是文件夹");
            return result;
        }
        File[] files = sourceDir.listFiles();
        //文件夹为空或者里面没有更多文件,辣鸡
        if (files == null || files.length == 0) {
            result.setResult(false);
            result.setMessage("没有找到录音文件");
            return result;
        } else if (files.length == 1) {
            //只有一个文件,美滋滋
            result.setResult(true);
            result.setMessage("");
            result.setAbsolutePath(files[0].getAbsolutePath());
            return result;
        } else {
            //目标录音文件
            File desAmrFile = new File(rootDir, String.valueOf("desAmr.amr"));
            if (!desAmrFile.createNewFile()) {
                result.setResult(false);
                result.setMessage("创建保存目标文件失败");
                return result;
            }
            List<File> list = Arrays.asList(files);
            //自然排序。时间戳靠前的排在前面
            Collections.sort(list);
            RandomAccessFile outRaf = new RandomAccessFile(desAmrFile, "rw");
            RandomAccessFile inRaf = null;
            final int length = list.size();
            //缓冲数据
            byte[] buff = new byte[BUFF_BYTE_ARRAY_LENGTH];
            int readLength = -1;
            for (int i = 0; i < length; i++) {
                //读取文件夹内容,只读形式打开
                inRaf = new RandomAccessFile(list.get(i), "r");
                if (i != 0) {
                    //跳过指定的字节以后再读
                    inRaf.seek(seekLength);
                    //循环读写
                    while ((readLength = inRaf.read(buff)) != -1) {
                        outRaf.write(buff, 0, readLength);
                    }
                } else {
                    //循环读写,
                    while ((readLength = inRaf.read(buff)) != -1) {
                        outRaf.write(buff, 0, readLength);
                    }
                }
                inRaf.close();
            }
            outRaf.close();
            result.setResult(true);
            result.setAbsolutePath(desAmrFile.getAbsolutePath());
            return result;
        }
    }

    /**
     * 合并结果的返回类
     */
    public static class Result implements Serializable {

        private static final long serialVersionUID = 1L;

        /**
         * true是合并成功,false是失败
         */
        private boolean result;

        /**
         * 合并的录音文件的路径
         */
        private String absolutePath;

        /**
         * 合并结果的信息,{@link #result}为true该值为空,为false会附带着失败原因
         */
        private String message;

        /**
         * 录音文件的长度
         */
        private int duration;

        public boolean isResult() {
            return result;
        }

        public void setResult(boolean result) {
            this.result = result;
        }

        public String getAbsolutePath() {
            return absolutePath;
        }

        public void setAbsolutePath(String absolutePath) {
            this.absolutePath = absolutePath;
        }

        public String getMessage() {
            return message;
        }

        public void setMessage(String message) {
            this.message = message;
        }

        public int getDuration() {
            return duration;
        }

        public void setDuration(int duration) {
            this.duration = duration;
        }

        @Override
        public String toString() {
            return "Result{" +
                    "result=" + result +
                    ", absolutePath='" + absolutePath + '\'' +
                    ", message='" + message + '\'' +
                    ", duration=" + duration +
                    '}';
        }
    }

}

虽说这个功能实现了,产品验收也没说有啥不妥。但是自己在自测的时候总觉得可以把录音的音量的音量调节大一点就好了。

放大音量

MediaRecorder是Google对底层录音功能封装的比较完善的一个类,其录制的音频文件格式并没有太多进行处理的机会了。在使用MediaRecorder的api的时候,我们就能发现,相对于AudioRecord,MediaRecorder的初始化以及配置使用都相对于比较简单。
之前开发录音功能的时间比较赶选择了MediaRecorder,闲下来学习实现了放大录音文件的音量的功能。

sample

前面讲了这么多,先把项目的sample放出来。
在sample里面已经实现了AudioRecord录音的暂停、重新开始录制,pcm2wav,wav2amr的功能。

思路

  1. 如何实现录音的暂停录音、继续录音

AudioRecord在录音的时候,只需要传入一个写入录音字节流的文件,start或者stop的时候,只要不reset或者release便可以实现文件的暂停录制和重新录制。

  1. 如何放大录音文件的录入音量

AudioRecord在录制的时候,需要初始化录音的采样频率、声道、位数配置等。这三个参数直接决定了录制文件的质量。在该文章的sample中,采用了8kHz的采样频率,单声道以及16bit的位数配置(opencore-amr库在不引入vo-amrwbenc支持的情况下,只支持8kHz采样的wav文件解析,所以sample的采样率是8kHz)。
在录音的时候,想要放大声音质量,只要将读到的字节流数组的元素逐个加倍放大即可。但是这里需要注意byte值的溢出,还有可能出现一些噪音问题。所以建议倍数不要设置过大,我这边采用了2倍的增益。

  1. pcm2wav,wav2amr

录音结束的时候,AudioRecord保存的文件是pcm的格式。想要保存成wav的格式,还需要加入wav特有的头文件。在Sample的Pcm2wav.java类中有具体实现。而将wav文件转换成amr文件,则需要使用opencore-amr库来编解码。项目路径,使用ndk编译以后即可完成wav2amr的功能。

感谢以下文章作者的无私分享,AMR编解码库的实现

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容

  • 前言: 记载资料多为网络搜集,侵删。 根据最近接触的整机项目做了一些整机音频相关基础知识的总结,如有不足或表述问题...
    Gawain_Knowknow阅读 8,137评论 0 4
  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,979评论 3 119
  • 视频直播的流程可以分为如下几步: 采集 —>处理—>编码和封装—>推流到服务器—>服务器流分发—>播放器流播放 1...
    韩瞅瞅阅读 4,379评论 0 12
  • 研究录音是源于即时通讯的项目。写出一个即时通讯很简单,但是写好一个即时通讯就不是一件容易的事,比如聊天中语音的加入...
    2c2z0阅读 2,362评论 0 2
  • 手机一般都有麦克风和摄像头,而Android系统就可以利用这些硬件来录制音视频了。为了增加对录制音视频的支持,An...
    安仔夏天勤奋阅读 9,045评论 4 22