首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >「Android音视频编码那点破事」第五章,使用MediaCodec编码AAC音频数据

「Android音视频编码那点破事」第五章,使用MediaCodec编码AAC音频数据

作者头像
阿利民
发布于 2022-05-16 04:42:07
发布于 2022-05-16 04:42:07
58600
代码可运行
举报
文章被收录于专栏:阿利民阿利民
运行总次数:0
代码可运行

封面出自:板栗懒得很

本章仅对部分代码进行讲解,以帮助读者更好的理解章节内容。 本系列文章涉及的项目HardwareVideoCodec已经开源到Github,支持软编和硬编。使用它你可以很容易的实现任何分辨率的视频编码,无需关心摄像头预览大小。一切都如此简单。目前已迭代多个稳定版本,欢迎查阅学习和使用,如有BUG或建议,欢迎Issue。

  在上一章我们讲到了MediaCodec的工作流程,以及如何利用MediaCodec进行H264编码。这一章的内容同样是MediaCodec,只不过是编码音频为AAC,整个流程大同小异。   上一章我们利用MediaCodec编码视频时,使用了Surface,所以可以不直接操作输入缓冲区队列。但是编码音频的时候,由于无法使用Surface,所以需要直接操作输入缓冲区队列。   这里我们需要通过AudioRecord采集PCM数据,然后把采集到的数据送进编码器进行编码。所以首先我们要初始化一个AudioRecord对象。   要使用录音,需要申请录音权限。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<uses-permission android:name="android.permission.RECORD_AUDIO"/>

  然后初始化AudioRecorder对象,初始化完成后就可以开始录制音频了。当然,这些操作都需要在子线程中进行。最后通过循环不停的从AudioRecorder中读取PCM数据,并通过回调把PCM数据发送给MediaCodec进行编码。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 初始化AudioRecord对象
 */
private fun config(){
    /**
     * 计算缓存PCM数据的Buffer最小大小
     * parameter.audio.sampleRateInHz = 16000
     * parameter.audio.pcm = AudioFormat.ENCODING_PCM_16BIT
     * parameter.audio.samplePerFrame = 1024
     * parameter.video.fps = 30
     *
     */
    val minBufferSize = AudioRecord.getMinBufferSize(parameter.audio.sampleRateInHz,
        AudioFormat.CHANNEL_IN_MONO, parameter.audio.pcm)
    /**
     * 计算buffer大小
     */
    bufferSize = parameter.audio.samplePerFrame * parameter.video.fps
    if (bufferSize < minBufferSize)
        bufferSize = (minBufferSize / parameter.audio.samplePerFrame + 1) * parameter.audio.samplePerFrame * 2
    
    debug_e("bufferSize: $bufferSize")
    /**
     * 新建储存PCM数据发Buffer
     */
    buffer = ByteArray(parameter.audio.samplePerFrame)
    /**
     * 新建AudioRecord对象
     */
    record = AudioRecord(MediaRecorder.AudioSource.MIC, parameter.audio.sampleRateInHz,
        AudioFormat.CHANNEL_IN_MONO, parameter.audio.pcm, bufferSize)
    /**
     * 开始录制音频
     */
    record?.startRecording()
}
private fun read() {
    /**
     * 读取PCM数据
     */
    val bufferReadResult = record!!.read(buffer, 0, parameter.audio.samplePerFrame)
    onPCMListener?.onPCMSample(buffer!!)
}

override fun run() {
    while (mStart) {
        read()
    }
}

  在正确拿到PCM数据后,就可以用MediaCodec进行编码了。我们先创建一个编码器格式对象,用来配置MediaCodec。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
fun createAudioFormat(parameter: Parameter, ignoreDevice: Boolean = false): MediaFormat? {
    val mediaFormat = MediaFormat()
    /**
     * 编码格式AAC:parameter.audio.mime = "audio/mp4a-latm"
     * 声道数量:parameter.audio.channel = 1
     * 频率:parameter.audio.sampleRateInHz = 16000
     * 码率:parameter.audio.bitrate = 64000
     * Level:parameter.audio.profile = MediaCodecInfo.CodecProfileLevel.AACObjectLC
     */
    mediaFormat.setString(MediaFormat.KEY_MIME, parameter.audio.mime)
    mediaFormat.setInteger(MediaFormat.KEY_CHANNEL_COUNT, parameter.audio.channel)
    mediaFormat.setInteger(MediaFormat.KEY_SAMPLE_RATE, parameter.audio.sampleRateInHz)
    mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, parameter.audio.bitrate)
    mediaFormat.setInteger(MediaFormat.KEY_AAC_PROFILE, parameter.audio.profile)
    return mediaFormat
}

  有了MediaFormat后,我们就可以开始创建编码器了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private fun initCodec() {
    val format = CodecHelper.createAudioFormat(parameter)
    try {
        codec = MediaCodec.createEncoderByType(format?.getString(MediaFormat.KEY_MIME))
        codec?.configure(format, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE)
        codec?.start()
        audioWrapper = AudioRecordWrapper(parameter)
        audioWrapper?.setOnPCMListener(this)
    } catch (e: Exception) {
        debug_e("Can not create codec")
    } finally {
        if (null == codec)
            debug_e("Can not create codec")
    }
}

  初始化之后通过OnPCMListener回调接收上文返回的PCM数据,并送入MediaCodec进行编码。最后通过循环从编码器输出缓冲区中拿出AAC数据。这里通过回调把AAC数据送进MediaMuxer进行音视频混合,最后生成mp4文件。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/**
 * 把PCM数据送入编码器的输入缓存队列
 */
private fun encode(buffer: ByteArray) {
    try {
        pTimer.record()
        /**
         * 获取输入缓存队列
         */
        inputBuffers = codec!!.inputBuffers
        /**
         * 输入输出缓存队列
         */
        outputBuffers = codec!!.outputBuffers
        /**
         * 从编码器中获取一个缓冲区的下标
         */
        val inputBufferIndex = codec!!.dequeueInputBuffer(WAIT_TIME)
        if (inputBufferIndex >= 0) {
            /**
             * 通过下标获取缓冲区
             */
            val inputBuffer = inputBuffers!![inputBufferIndex]
            inputBuffer.clear()
            /**
             * 把PCM数据送入缓冲区
             */
            inputBuffer.put(buffer)
            /**
             * 把带有PCM的数据缓冲区送进编码器
             */
            codec!!.queueInputBuffer(inputBufferIndex, 0, buffer.size, 0, 0)
        }
        /**
         * 从编码器中获取编码后的数据
         */
        dequeue()
    } catch (e: Exception) {
        e.printStackTrace()
    }
}
/**
 * 从编码器中获取编码后的数据
 */
private fun dequeue(): Boolean {
    try {
        /**
         * 从输出缓冲区取出一个Buffer,返回一个状态
         * 这是一个同步操作,所以我们需要给定最大等待时间WAIT_TIME,一般设置为10000ms
         */
        val flag = codec!!.dequeueOutputBuffer(bufferInfo, WAIT_TIME)
        when (flag) {
            MediaCodec.INFO_TRY_AGAIN_LATER -> {//等待超时,需要再次等待,通常忽略
                return false
            }
            /**
             * 输出格式改变,很重要
             * 这里必须把outputFormat设置给MediaMuxer,而不能不能用inputFormat代替,它们时不一样的,不然无法正确生成mp4文件
             */
            MediaCodec.INFO_OUTPUT_FORMAT_CHANGED -> {
                debug_v("AUDIO INFO_OUTPUT_FORMAT_CHANGED")
                onSampleListener?.onFormatChanged(codec!!.outputFormat)
            }
            else -> {
                if (flag < 0) return@dequeue false//如果小于零,则跳过
                val data = codec!!.outputBuffers[flag]//否则代表编码成功,可以从输出缓冲区队列取出数据
                if (null != data) {
                    val endOfStream = bufferInfo.flags and MediaCodec.BUFFER_FLAG_END_OF_STREAM
                    if (endOfStream == 0) {//如果没有收到BUFFER_FLAG_END_OF_STREAM信号,则代表输出数据时有效的
                        bufferInfo.presentationTimeUs = pTimer.presentationTimeUs
                        //通过回调,把编码后的数据送进MediaMuxer
                        onSampleListener?.onSample(bufferInfo, data)
                    }
                    //缓冲区使用完后必须把它还给MediaCodec,以便再次使用,至此一个流程结束,再次循环
                    codec!!.releaseOutputBuffer(flag, false)
//                        if (endOfStream == MediaCodec.BUFFER_FLAG_END_OF_STREAM) {
//                            return true
//                        }
                    return true
                }
            }
        }
    } catch (e: Exception) {
        e.printStackTrace()
    }
    return false
}

  以上就是本章关于MediaCodec编码PCM的全部学习内容,比较简单,关于MediaCodec的使用在第四章已经有了很详细的讲解,使用MediaCodec编码音视频的流程都是一样的。如果理解还不够透彻,欢迎查阅学习第四章。如果有疑问或者错误,欢迎在评论区留言。

本章知识点:

  1. 使用MediaCodec进行AAC编码。

本章相关源码·HardwareVideoCodec项目:

  • CodecHelper
  • AudioRecordWrapper
  • AudioEncoderImpl

分类:

多媒体系列文章

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-10-21,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 阿利民 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
「Android音视频编码那点破事」第四章,使用MediaCodec实现H264编码
  说到Android的视频硬编码,很多新人首先会想到MediaRecorder,这可以说是Android早期版本视频硬编码的唯一选择。这个类的使用很简单,只需要给定一个Surface(输入)和一个File(输出),它就给你生成一个标准的mp4文件。   但越是简单的东西便意味着越难以控制,MediaRecorder的缺点很明显。相信很多人在接触到断点视频录制这个需求的时候,首先会想到使用MediaRecorder,很遗憾,这个东西并不能给你很多期待,就像一开始的我一样。   首先,MediaRecorder并没有断点录制的API,当然你可以使用一些“小技巧”,每次录制的时候,都把MediaRecorder stop掉,然后再次初始化,这样就会生成一系列的视频,最后把它们拼接起来。然而问题在于,每次初始化MediaRecorder都需要消耗很长时间,这意味着,当用户快速点击录制按钮的时候可能会出现问题。对于这个问题,你可以等到MediaRecorder初始化完成才让用户点击开始录制,但是这样往往会因为等待时间过长,导致用户体验极差。   这种情况下,一个可控的视频编码器是必须的。虽然在Android 4.4以前我们没得选择,但是在Android 4.4之后,我们有了MediaCodec,一个完全可控的视频编码器,虽然无法直接输出mp4(需要配合MediaMuxer来对音视频进行混合,最终输出mp4,或者其它封装格式)。如今的Android生态,大部分手机都已经是Android 5.0系统,完全可以使用MediaCodec来进行音视频编码的开发,而MediaRecorder则降级作为一个提高兼容性的备选方案。   废话不多说,我们直接步入正题。要想正确的使用MediaCodec,我们首先得先了解它的工作流程,关于这个,强烈大家去看一下Android文档。呃呃,相信在这个快速开发为王道的环境,没几个人会去看,所以还是在这里简单介绍一下。
阿利民
2022/05/16
9870
「Android音视频编码那点破事」第四章,使用MediaCodec实现H264编码
一文掌握直播技术:实时音视频采集、编码、传输与播放
从游戏、教育、电商到娱乐,直播技术的应用场景无处不在。随着移动端的网速越来越快,直播技术的普及和发展将更加迅速。
陆业聪
2024/08/19
1.6K0
一文掌握直播技术:实时音视频采集、编码、传输与播放
Android AVDemo(2):音频编码,采集 PCM 数据编码为 AAC丨音视频工程示例
iOS/Android 客户端开发同学如果想要开始学习音视频开发,最丝滑的方式是对音视频基础概念知识有一定了解后,再借助 iOS/Android 平台的音视频能力上手去实践音视频的采集 → 编码 → 封装 → 解封装 → 解码 → 渲染过程,并借助音视频工具来分析和理解对应的音视频数据。
关键帧
2022/06/13
1.1K0
Android AVDemo(2):音频编码,采集 PCM 数据编码为 AAC丨音视频工程示例
MediaCodec录制音视频并将合成为一个文件
音频录制 相关参考 MediaCodec硬编码pcm2aac 主要分为以下几步骤:
曾大稳
2020/01/20
2.4K0
MediaCodec录制音视频并将合成为一个文件
Android原生编解码接口 MediaCodec 之——踩坑
MediaCodec 有两种方式触发输出关键帧,一是由配置时设置的 KEY_FRAME_RATE 和KEY_I_FRAME_INTERVAL参数自动触发,二是运行过程中通过 setParameters 手动触发输出关键帧。
全栈程序员站长
2022/11/01
7K1
Android原生编解码接口 MediaCodec 之——踩坑
MediaCodec进行AAC编解码(AudioRecord采集录音)
最近工作比较忙,很久没有更新这个系列的文章。我们先回顾一下上一篇MediaCodec进行AAC编解码(文件格式转换)的内容,里面介绍了MediaExtractor的使用,MediaCodec进行音频文件的解码和编码,ADTS的介绍和封装。今天这篇文章在此基础上跟大家一起学习如何通过Android设备进行音频的采集,然后使用MediaCodec进行AAC编码,最后输出到文件。这部分我们关注的重点就是在如何进行音频的采集。 项目代码github对应的代码版本v1.7。大家一定要注意下载对应的代码版本调试。
用户2929716
2018/08/23
2K0
MediaCodec进行AAC编解码(AudioRecord采集录音)
Android MediaCodec 硬编码 H264 文件
在 Android 4.1 版本提供了 MediaCodec 接口来访问设备的编解码器,不同于 FFmpeg 的软件编解码,它采用的是硬件编解码能力,因此在速度上会比软解更具有优势,但是由于 Android 的碎片化问题,机型众多,版本各异,导致 MediaCodec 在机型兼容性上需要花精力去适配,并且编解码流程不可控,全交由厂商的底层硬件去实现,最终得到的视频质量不一定很理想。
音视频开发进阶
2019/11/18
3.6K0
MediaCodec硬编码pcm2aac
MediaCodec是Android(api>=16)提供的一个多媒体硬解编码库,能实现音视频的编解码。
曾大稳
2018/09/11
1.5K0
【Android 音视频开发打怪升级:OpenGL渲染视频画面篇】六、Android音视频硬编码:生成一个MP4
在【音视频硬解码流程:封装基础解码框架】这篇文章中,介绍了如何使用Android原生提供的硬编解码工具MediaCodec,对视频进行解码。同时,MediaCodec也可以实现对音视频的硬编码。
开发的猫
2020/04/02
2.3K1
【Android 音视频开发打怪升级:OpenGL渲染视频画面篇】六、Android音视频硬编码:生成一个MP4
【Android 音视频开发打怪升级:音视频硬解码篇】三、音视频播放:音视频同步
在上一篇文章定义的解码流程框架基类中,预留了几个虚函数,留给子类初始化自己的东西,本篇,就来看看如何实现。
开发的猫
2020/04/01
2.6K0
【Android 音视频开发打怪升级:音视频硬解码篇】三、音视频播放:音视频同步
短视频源码开发,Android端短视频功能的快速实现
一、短视频内容生产 优质短视频内容的产生依赖于短视频的采集和特效编辑,这就要求在进行短视频源码开发时,用到基础的美颜、混音、滤镜、变速、图片视频混剪、字幕等功能,在这些功能基础上,进行预处理,结合OpenGL、AI、AR技术,产生很多有趣的动态贴纸玩法,使得短视频内容更具创意。
云豹科技程序员
2021/06/01
1.2K0
干货 | 移动端使用OpenGL转场特效的音视频合成应用
近年来短视频的火爆,让内容创作类的APP获得了巨大的流量。用户通过这类工具编辑自己的短视频,添加各式各样的炫酷特效,从而呈现出更加丰富多彩的视频内容。本文将会介绍如何使用移动端原生API,将图片添加转场特效并且最终合成为视频的基本流程。
携程技术
2023/11/11
7570
干货 | 移动端使用OpenGL转场特效的音视频合成应用
音视频入门之音频采集、编码、播放
今天我们学习音频的采集、编码、生成文件、转码等操作,我们生成三种格式的文件格式,pcm、wav、aac 三种格式,并且我们用 AudioStack 来播放音频,最后我们播放这个音频。
Android技术干货分享
2019/04/18
3.9K0
Android AVDemo(5):音频解码,免费获得源码丨音视频工程示例
iOS/Android 客户端开发同学如果想要开始学习音视频开发,最丝滑的方式是对音视频基础概念知识有一定了解后,再借助 iOS/Android 平台的音视频能力上手去实践音视频的采集 → 编码 → 封装 → 解封装 → 解码 → 渲染过程,并借助音视频工具来分析和理解对应的音视频数据。
关键帧
2022/06/13
5410
Android AVDemo(5):音频解码,免费获得源码丨音视频工程示例
刷抖音上瘾后,决定探究如果做一款类似抖音短视频app
优质短视频内容的产生依赖于短视频的采集和特效编辑,这就要求在进行抖音APP开发时,用到基础的美颜、混音、滤镜、变速、图片视频混剪、字幕等功能,在这些功能基础上,进行预处理,结合OpenGL、AI、AR技术,产生很多有趣的动态贴纸玩法,使得短视频内容更具创意。
Android技术干货分享
2019/06/17
2.6K2
Android 获取视频缩略图(获取视频每帧数据)的优化方案
注意:如果不缩小图片的话,建议还是使用MediaMetadataRetriever。 使用当前库的话,调用metadataRetriever.forceFallBack(true);
deep_sadness
2019/01/28
5.1K0
Android MediaCodec 使用说明
最近公司要求提供一个支持 Android 硬件转码的底层库,所以自己从头去看了 MediaCodec 相关的知识,费了老大的劲终于完成了。
字节流动
2021/03/16
1.9K0
Android MediaCodec 使用说明
Android音视频硬编码与混合(三)
硬编码:使用非CPU进行编码,如显卡GPU、专用的DSP、FPGA、ASIC芯片等
PengJie
2021/01/03
2.7K0
Android 音频开发入门指南
Android 平台提供了一套丰富的音频 API,使得开发者可以轻松地为应用添加音频播放、录制、处理等功能。这些 API 包括:
陆业聪
2024/07/23
9670
Android 音频开发入门指南
使用 MediaExtractor 及 MediaCodec 解码音视频
大家好,本文是 iOS/Android 音视频专题 的第四篇,从本篇文章开始我们将动手编写代码。代码工程将在 Github 进行托管。
100001509164
2022/01/20
2.8K0
使用 MediaExtractor 及 MediaCodec 解码音视频
推荐阅读
相关推荐
「Android音视频编码那点破事」第四章,使用MediaCodec实现H264编码
更多 >
领券
一站式MCP教程库,解锁AI应用新玩法
涵盖代码开发、场景应用、自动测试全流程,助你从零构建专属AI助手
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档