欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android錄音并且輸出為Mp4文件的方法教程

 更新時(shí)間:2018年08月03日 11:43:11   作者:滑板上的老砒霜  
這篇文章主要給大家介紹了關(guān)于Android錄音并且輸出為Mp4文件的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

前言

錄音采用的是AudioRecord,通過(guò)MediaCodec進(jìn)行編碼,用MediaMuxer合成輸出MP4文件。

1.

這里用AudioRecord來(lái)得到從麥克風(fēng)錄制的聲音,AudiorRecord的用法還是比較簡(jiǎn)單的,首先初始化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的構(gòu)造函數(shù)需要出入幾個(gè)參數(shù)。

(1).

audioSource代表音頻來(lái)源,這里傳入MediaRecorder.AudioSource.MIC,代表音頻來(lái)源于麥克風(fēng)。

(2).

sampleRateInHz代表以赫茲表示的采樣率,傳入44100,這個(gè)數(shù)值可以保證所有設(shè)備都正常工作。

(3).

channelConfig代表聲道配置,AudioFormat.CHANNEL_IN_STEREO代表傳入立體聲。

(4).

audioFormat代表音頻數(shù)據(jù)將被返回的格式。傳入AudioFormat.ENCODING_PCM_16BIT。

(5).

bufferSizeInBytes寫入音頻數(shù)據(jù)的緩沖區(qū)的總大?。ㄒ宰止?jié)為單位)這里默認(rèn)傳入getMinBufferSize,這個(gè)方法返回成功創(chuàng)建AudioRecord實(shí)例所需的緩沖區(qū)大小的最小值。

開始錄音時(shí),啟動(dòng)一個(gè)線程

private val recordRunnable = Runnable {


 val data = ByteArray(minSize)
 //AudioRecord開始錄音
 audioRecord?.startRecording()
 while (isRecording)
 {
  //將音頻數(shù)據(jù)寫入ByteArray
  audioRecord?.read(data, 0, data.size)
  audioEncoder.start()
  audioEncoder.drainEncoder(data)
 }
 audioEncoder.release()
 audioRecord?.stop()
 audioRecord?.release()
 audioRecord = null
}

這樣就把音頻數(shù)據(jù)寫入一個(gè)bytearray,然后將數(shù)據(jù)傳入AudioEncoder進(jìn)行編碼輸出。

2.

AudioRecord得到的音頻數(shù)據(jù)格式是pcm的,一般情況下無(wú)法bofang(可以通過(guò)AudioTrack播放),所以我們需要一次編碼轉(zhuǎn)換,這里用到的就是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)
}

這里需要?jiǎng)?chuàng)建一個(gè)MediaFormat,具體需要傳入的參數(shù)大家可以參考開發(fā)者文檔,需要注意的是音頻和視頻的MediaFormat設(shè)置的參數(shù)是不一樣的,接著調(diào)用MediaCodec的configure,此時(shí)MediaCodec已經(jīng)進(jìn)入了configured的狀態(tài),可以開始進(jìn)行編碼了。

這里說(shuō)到MediaCodec的狀態(tài),大家可以看來(lái)自開發(fā)者文檔的MediaCodec的狀態(tài)機(jī)圖片。


看一下MediaCodec的工作過(guò)程:

可以將MediaCodec理解為傳送帶,將空的buffers傳給audiorecord,audiorecord將得到的bytearray放入空的buffers,然后傳入MediaCodec,mediaCodec編碼后,傳入MediaMuxer,MediaMuxer寫入編碼后的數(shù)據(jù)再講buffers傳給MediaCodec,MediaCodec清空使用過(guò)的Buffers,再傳給AudioRecord。buffer是java nio庫(kù)里的類,這里就不詳述了,不清楚的請(qǐng)自行g(shù)oogle。

調(diào)用configure后,我們就進(jìn)入了configred狀態(tài),之后當(dāng)audiorecord得到數(shù)據(jù)后,當(dāng)MediaCodec調(diào)用start方法后,將ByteArray傳入MediaCodec,進(jìn)行編碼:

 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
  }
}

這個(gè)方法就對(duì)應(yīng)了MediaCodec對(duì)應(yīng)的工作過(guò)程。

3.

MediaMuxer用來(lái)合成并輸出音頻,MediaMuxer用法還是比較簡(jiǎn)單的,這里就不詳述了,需要注意的是,MediaMuxer只能合并一個(gè)音頻軌道和一個(gè)視頻軌道,還要注意的是要在addTrack調(diào)用之后再調(diào)用star方法。

最后附上項(xiàng)目地址

gitlab

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • 封裝的android監(jiān)聽手指左右滑動(dòng)屏幕的事件類分享

    封裝的android監(jiān)聽手指左右滑動(dòng)屏幕的事件類分享

    這篇文章主要介紹了封裝的android監(jiān)聽手指左右滑動(dòng)屏幕的事件類分享,本文分別給出了簡(jiǎn)單處理方法的代碼和封裝好的處理類代碼,需要的朋友可以參考下
    2015-05-05
  • Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn)

    Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn)

    本篇文章主要介紹了Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-04-04
  • Android ReboundScrollView仿IOS拖拽回彈效果

    Android ReboundScrollView仿IOS拖拽回彈效果

    這篇文章主要為大家詳細(xì)介紹了Android ReboundScrollView仿IOS拖拽回彈效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-11-11
  • android開發(fā)教程之系統(tǒng)資源的使用方法 android資源文件

    android開發(fā)教程之系統(tǒng)資源的使用方法 android資源文件

    這篇文章主要介紹了android中的系統(tǒng)資源的使用方法,包括顏色資源 、字符串資源、尺寸資源、XML資源文件,需要的朋友可以參考下
    2014-02-02
  • Android入門計(jì)算器編寫代碼

    Android入門計(jì)算器編寫代碼

    這篇文章主要為大家詳細(xì)介紹了Android入門計(jì)算器編寫代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-08-08
  • flutter ExpansionTile 層級(jí)菜單的實(shí)現(xiàn)

    flutter ExpansionTile 層級(jí)菜單的實(shí)現(xiàn)

    這篇文章主要介紹了flutter ExpansionTile 層級(jí)菜單的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-07-07
  • Android編程之非調(diào)用系統(tǒng)界面實(shí)現(xiàn)發(fā)送彩信的方法(MMS)

    Android編程之非調(diào)用系統(tǒng)界面實(shí)現(xiàn)發(fā)送彩信的方法(MMS)

    這篇文章主要介紹了Android編程之非調(diào)用系統(tǒng)界面實(shí)現(xiàn)發(fā)送彩信的方法,涉及Android源碼中的mms的使用技巧,需要的朋友可以參考下
    2016-01-01
  • Flutter Component動(dòng)畫的顯和隱最佳實(shí)踐

    Flutter Component動(dòng)畫的顯和隱最佳實(shí)踐

    這篇文章主要為大家介紹了Flutter Component動(dòng)畫的顯和隱最佳實(shí)踐詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03
  • Kotlin Channel處理多個(gè)數(shù)據(jù)組合的流

    Kotlin Channel處理多個(gè)數(shù)據(jù)組合的流

    最近項(xiàng)目中對(duì) kotlin 的使用比較多。不得不說(shuō) kotlin 確實(shí)可以極大的提高 android 的開發(fā)效率,channel用于協(xié)程之間的通訊,使用send和receive往通道里寫入或者讀取數(shù)據(jù),2個(gè)方法為非阻塞掛起函數(shù),channel是熱流,不管有沒(méi)有訂閱者都會(huì)發(fā)送
    2022-11-11
  • Android?Jetpack結(jié)構(gòu)運(yùn)用Compose實(shí)現(xiàn)微博長(zhǎng)按點(diǎn)贊彩虹效果

    Android?Jetpack結(jié)構(gòu)運(yùn)用Compose實(shí)現(xiàn)微博長(zhǎng)按點(diǎn)贊彩虹效果

    Compose在動(dòng)畫方面下足了功夫,提供了豐富的API。但也正由于API種類繁多,如果想一氣兒學(xué)下來(lái),最終可能會(huì)消化不良,導(dǎo)致似懂非懂。結(jié)合例子學(xué)習(xí)是一個(gè)不錯(cuò)的方法,本文就帶大家邊學(xué)邊做,通過(guò)實(shí)現(xiàn)一個(gè)微博長(zhǎng)按點(diǎn)贊的動(dòng)畫效果,學(xué)習(xí)了解Compose動(dòng)畫的常見思路和開發(fā)技巧
    2022-07-07

最新評(píng)論