Android录音并且输出为Mp4文件的方法教程

 更新时间:2018年08月03日 11:43:11   作者:滑板上的老砒霜  
这篇文章主要给大家介绍了关于Android录音并且输出为Mp4文件的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

录音采用的是AudioRecord,通过MediaCodec进行编码,用MediaMuxer合成输出MP4文件。

1.

这里用AudioRecord来得到从麦克风录制的声音,AudiorRecord的用法还是比较简单的,首先初始化AudioRecord

fun prepare(file: File?, outputFormat: Int = MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4, audioSource: Int = MediaRecorder.AudioSource.MIC, sampleRateInHz: Int = 44100, channelConfig: Int = AudioFormat.CHANNEL_IN_STEREO, audioFormat: Int = AudioFormat.ENCODING_PCM_16BIT, bufferSizeInBytes: Int = AudioRecord.getMinBufferSize(sampleRateInHz, channelConfig, audioFormat))
{
 //初始化AudioRecord
 prepareAudioRecord(audioSource, sampleRateInHz, channelConfig, audioFormat, bufferSizeInBytes)
 //初始化输出文件
 prepareOutputFile(file)
 //初始化AudioEncoder
 prepareAudioEncoder(sampleRateInHz, outputFormat)
}


private fun prepareAudioRecord(audioSource: Int, sampleRateInHz: Int, channelConfig: Int, audioFormat: Int, bufferSizeInBytes: Int)
{
 minSize = AudioRecord.getMinBufferSize(sampleRateInHz, channelConfig, audioFormat)

 audioRecord = AudioRecord(audioSource, sampleRateInHz, channelConfig, audioFormat, bufferSizeInBytes)
}

AudioRecord的构造函数需要出入几个参数。

(1).

audioSource代表音频来源,这里传入MediaRecorder.AudioSource.MIC,代表音频来源于麦克风。

(2).

sampleRateInHz代表以赫兹表示的采样率,传入44100,这个数值可以保证所有设备都正常工作。

(3).

channelConfig代表声道配置,AudioFormat.CHANNEL_IN_STEREO代表传入立体声。

(4).

audioFormat代表音频数据将被返回的格式。传入AudioFormat.ENCODING_PCM_16BIT。

(5).

bufferSizeInBytes写入音频数据的缓冲区的总大小(以字节为单位)这里默认传入getMinBufferSize,这个方法返回成功创建AudioRecord实例所需的缓冲区大小的最小值。

开始录音时,启动一个线程

private val recordRunnable = Runnable {


 val data = ByteArray(minSize)
 //AudioRecord开始录音
 audioRecord?.startRecording()
 while (isRecording)
 {
  //将音频数据写入ByteArray
  audioRecord?.read(data, 0, data.size)
  audioEncoder.start()
  audioEncoder.drainEncoder(data)
 }
 audioEncoder.release()
 audioRecord?.stop()
 audioRecord?.release()
 audioRecord = null
}

这样就把音频数据写入一个bytearray,然后将数据传入AudioEncoder进行编码输出。

2.

AudioRecord得到的音频数据格式是pcm的,一般情况下无法bofang(可以通过AudioTrack播放),所以我们需要一次编码转换,这里用到的就是MediaCodec,MediaCodec这里我封装在AudioEncoder里。
我们首先要初始化MediaCodec:

private fun prepareAudioCodec(bitrate: Int, sampleRate: Int)
{
 bufferInfo = MediaCodec.BufferInfo()
 val mediaFormat = MediaFormat()
 mediaFormat.setInteger(MediaFormat.KEY_AAC_PROFILE, MediaCodecInfo.CodecProfileLevel.AACObjectLC)
 mediaFormat.setString(MediaFormat.KEY_MIME, MediaFormat.MIMETYPE_AUDIO_AAC)
 mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, bitrate)
 mediaFormat.setInteger(MediaFormat.KEY_CHANNEL_COUNT, 2)
 mediaFormat.setInteger(MediaFormat.KEY_SAMPLE_RATE, sampleRate)
 audioCodec = MediaCodec.createEncoderByType(MediaFormat.MIMETYPE_AUDIO_AAC)
 audioCodec.configure(mediaFormat, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE)
}

这里需要创建一个MediaFormat,具体需要传入的参数大家可以参考开发者文档,需要注意的是音频和视频的MediaFormat设置的参数是不一样的,接着调用MediaCodec的configure,此时MediaCodec已经进入了configured的状态,可以开始进行编码了。

这里说到MediaCodec的状态,大家可以看来自开发者文档的MediaCodec的状态机图片。


看一下MediaCodec的工作过程:

可以将MediaCodec理解为传送带,将空的buffers传给audiorecord,audiorecord将得到的bytearray放入空的buffers,然后传入MediaCodec,mediaCodec编码后,传入MediaMuxer,MediaMuxer写入编码后的数据再讲buffers传给MediaCodec,MediaCodec清空使用过的Buffers,再传给AudioRecord。buffer是java nio库里的类,这里就不详述了,不清楚的请自行google。

调用configure后,我们就进入了configred状态,之后当audiorecord得到数据后,当MediaCodec调用start方法后,将ByteArray传入MediaCodec,进行编码:

 fun drainEncoder(data: ByteArray)
{

  val inIndex = audioCodec.dequeueInputBuffer(0)
  if (inIndex > 0)
  {
    val inBuffer = getInBuffer(inIndex)
    inBuffer.clear()
    inBuffer.put(data)
    if (!isEncoding)
    {
      audioCodec.queueInputBuffer(inIndex, 0, 0, System.nanoTime() / 1000, BUFFER_FLAG_END_OF_STREAM)

    } else
    {
      audioCodec.queueInputBuffer(inIndex, 0, data.size, System.nanoTime() / 1000, 0)
    }

  }

  do
  {
    val outIndex = audioCodec.dequeueOutputBuffer(bufferInfo, 0)
    when
    {
      outIndex > 0 ->
      {

        if (bufferInfo.size != 0)
        {
          val outBuffer = getOutBuffer(outIndex)
          outBuffer.position(bufferInfo.offset)
          outBuffer.limit(bufferInfo.offset + bufferInfo.size)
          mediaMuxer.writeSampleData(trackIndex, outBuffer, bufferInfo)
        }
        audioCodec.releaseOutputBuffer(outIndex, false)
      }
      outIndex == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED ->
      {
        trackIndex = mediaMuxer.addTrack(audioCodec.outputFormat)
        mediaMuxer.start()
      }

    }
  } while (outIndex > 0)

  if (bufferInfo.flags and MediaCodec.BUFFER_FLAG_END_OF_STREAM != 0)
  {

    isEncoding = false
  }
}

这个方法就对应了MediaCodec对应的工作过程。

3.

MediaMuxer用来合成并输出音频,MediaMuxer用法还是比较简单的,这里就不详述了,需要注意的是,MediaMuxer只能合并一个音频轨道和一个视频轨道,还要注意的是要在addTrack调用之后再调用star方法。

最后附上项目地址

gitlab

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

相关文章

  • android sdk安装及开发环境部署

    android sdk安装及开发环境部署

    本文给大家详细讲解了android sdk安装方法以及android开发环境部署方法,非常的细致全面,有需要的小伙伴务必详细研究下。
    2015-11-11
  • Android实现省市区三级联动

    Android实现省市区三级联动

    这篇文章主要为大家详细介绍了Android实现省市区三级联动,Spinner实现省市区的三级联动,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • android下拉刷新ListView的介绍和实现代码

    android下拉刷新ListView的介绍和实现代码

    在当下,列表组件不带下拉刷新的都不好意思叫列表。第一次完成列表的下拉刷新功能的时候,直接在Activity中实现,虽然功能上是实现了,总体上感觉很乱。所以第二次用到的时候,就想着封装成一个组件,实现和Activity的解耦。
    2013-04-04
  • Android列表点击事件定义的一些思考

    Android列表点击事件定义的一些思考

    大家好,本篇文章主要讲的是Android列表点击事件定义的一些思考,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • Android使用ViewStub实现布局优化方法示例

    Android使用ViewStub实现布局优化方法示例

    这篇文章主要为大家介绍了Android使用ViewStub实现布局优化方法示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • Android编程中避免内存泄露的方法总结

    Android编程中避免内存泄露的方法总结

    这篇文章主要介绍了Android编程中避免内存泄露的方法总结,本文讲解了最可能造成内存泄露的几个点,并总结出如何应对这些内存泄露,需要的朋友可以参考下
    2014-08-08
  • Android App开发的自动化测试框架UI Automator使用教程

    Android App开发的自动化测试框架UI Automator使用教程

    UI Automator为Android程序的UI开发提供了测试环境,这里我们就来看一下Android App开发的自动化测试框架UI Automator使用教程,需要的朋友可以参考下
    2016-07-07
  • adnroid已安装应用中检测某应用是否安装的代码实例

    adnroid已安装应用中检测某应用是否安装的代码实例

    这篇文章主要介绍了Android怎么检测一个应用是否安装的方法,大家参考使用吧
    2013-11-11
  • Android Studio一直处于Building的两种解决方法

    Android Studio一直处于Building的两种解决方法

    很多朋友都遇到过打开别人的项目一直处于Building‘XXX’Gradle project info的情况。下面小编给大家带来了Android Studio一直处于Building的解决方法,感兴趣的朋友一起看看吧
    2018-08-08
  • android 9PNG图片制作过程(图文介绍)

    android 9PNG图片制作过程(图文介绍)

    我们想要是有些图片可以拉伸而不失真多好啊,这时候我们就要想起android为我们提供的9.png格式的图片了,9.png格式的图片是安卓平台上新创的一种被拉伸却不失真的玩意
    2013-01-01

最新评论