AudioTrack播放acc格式音頻

AudioTrack本身只支持播放pcm格式音頻,想要使用AudioTrack播放acc格式音頻,還需要其他api來進行數據封裝。

MediaExtractor

MediaCodec

package com.zero.demo;

/**
* Created by yingkun_che on 19-6-19.
* 使用AudioTrack 播放acc音頻測試
*/
import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioTrack;
import android.media.MediaCodec;
import android.media.MediaCodec.BufferInfo;
import android.media.MediaExtractor;
import android.media.MediaFormat;
import android.util.Log;
import java.io.IOException;
import java.nio.ByteBuffer;


public class AudioDecoderThread {
    private static final int TIMEOUT_US = 1000;
    private MediaExtractor mExtractor;
    private MediaCodec mDecoder;


    private boolean eosReceived;
    private int mSampleRate = 0;


    private final String TAG = "AACPlay";
    String accPath="/storage/emulated/0/Android/data/com.zero.demo/files/zero_editor_kuato/resources/themes/satura/Stratosphere.aac";
    public void startPlay(String path) throws IOException {
        eosReceived = false;
        //創建MediaExtractor對象用來解AAC封裝
        mExtractor = new MediaExtractor();
        try {
            //設置需要MediaExtractor解析的文件的路徑
            mExtractor.setDataSource(path);
        } catch (IOException e) {
            e.printStackTrace();
        }

        MediaFormat format = mExtractor.getTrackFormat(0);
        if (format == null)
        {
            Log.e(TAG,"format is null");
            return;
        }

        //判斷當前幀的文件類型是否爲audio
        String mime = format.getString(MediaFormat.KEY_MIME);
        if (mime.startsWith("audio/")) {
            Log.d(TAG, "format : " + format);
            //獲取當前幀的採樣率
            mExtractor.selectTrack(0);
            mSampleRate = format.getInteger(MediaFormat.KEY_SAMPLE_RATE);
            //獲取當前幀的通道數
            int channel  = format.getInteger(MediaFormat.KEY_CHANNEL_COUNT);
            //音頻文件長度
            long duration = format.getLong(MediaFormat.KEY_DURATION);
            Log.d(TAG,"length:"+duration/1000000);
        }

        //創建MediaCodec對象
        mDecoder = MediaCodec.createDecoderByType(mime);
        //配置MediaCodec
        mDecoder.configure(format, null, null, 0);

        if (mDecoder == null) {
            Log.e(TAG, "Can't find video info!");
            return;
        }
        //啓動MediaCodec
        mDecoder.start();
        new Thread(AACDecoderAndPlayRunnable).start();
    }

    Runnable AACDecoderAndPlayRunnable = new Runnable() {
        @Override
        public void run() {
            AACDecoderAndPlay();
        }
    };

    private void AACDecoderAndPlay() {
        ByteBuffer[] inputBuffers = mDecoder.getInputBuffers();
        ByteBuffer[] outputBuffers = mDecoder.getOutputBuffers();

        BufferInfo info = new BufferInfo();

        int buffsize = AudioTrack.getMinBufferSize(mSampleRate, AudioFormat.CHANNEL_OUT_STEREO, AudioFormat.ENCODING_PCM_16BIT);
        // 創建AudioTrack對象
        AudioTrack audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, mSampleRate,
                AudioFormat.CHANNEL_OUT_STEREO,
                AudioFormat.ENCODING_PCM_16BIT,
                buffsize,
                AudioTrack.MODE_STREAM);
        //啓動AudioTrack
        audioTrack.play();
    int num=0;
        while (!eosReceived) {
            int inIndex = mDecoder.dequeueInputBuffer(TIMEOUT_US);
            if (inIndex >= 0) {
                ByteBuffer buffer = inputBuffers[inIndex];
                //從MediaExtractor中讀取一幀待解數據
                int sampleSize = mExtractor.readSampleData(buffer, 0);
                if (sampleSize < 0) {
                    // We shouldn't stop the playback at this point, just pass the EOS
                    // flag to mDecoder, we will get it again from the
                    // dequeueOutputBuffer
                    Log.d(TAG, "InputBuffer BUFFER_FLAG_END_OF_STREAM");
                    mDecoder.queueInputBuffer(inIndex, 0, 0, 0, MediaCodec.BUFFER_FLAG_END_OF_STREAM);


                } else {
                    //向MediaDecoder輸入一幀待解碼數據
                    mDecoder.queueInputBuffer(inIndex, 0, sampleSize, mExtractor.getSampleTime(), 0);
                    mExtractor.advance();
                }
                //從MediaDecoder隊列取出一幀解碼後的數據
                int outIndex = mDecoder.dequeueOutputBuffer(info, TIMEOUT_US);
                if (info.size>0){
                    Log.d(TAG, "INFO_OUTPUT_BUFFERS_CHANGED num: "+num);
                    num=0;
                }else{
                    num++;
                }
                switch (outIndex) {
                    case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED:
                        Log.d(TAG, "INFO_OUTPUT_BUFFERS_CHANGED");
                        outputBuffers = mDecoder.getOutputBuffers();
                        break;
                    case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED:
                        MediaFormat format = mDecoder.getOutputFormat();
                        Log.d(TAG, "New format " + format);
                        audioTrack.setPlaybackRate(format.getInteger(MediaFormat.KEY_SAMPLE_RATE));
                        break;
                    case MediaCodec.INFO_TRY_AGAIN_LATER:
                        Log.d(TAG, "dequeueOutputBuffer timed out!");
                        break;
                    default:
                        ByteBuffer outBuffer = outputBuffers[outIndex];
                        //Log.v(TAG, "outBuffer: " + outBuffer);

                        final byte[] chunk = new byte[info.size];
                        // Read the buffer all at once
                        outBuffer.get(chunk);
                        //清空buffer,否則下一次得到的還會得到同樣的buffer
                        outBuffer.clear();
                        // AudioTrack write data
                        audioTrack.write(chunk, info.offset, info.offset + info.size);
                        mDecoder.releaseOutputBuffer(outIndex, false);//這一行必須加,因爲在P上mDecoder.dequeueOutputBuffer 前面會有80次沒有數據,需要釋放緩衝區,否則無聲音
                        break;
                }
                // 所有幀都解碼、播放完之後退出循環
                if ((info.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0) {
                    Log.d(TAG, "OutputBuffer BUFFER_FLAG_END_OF_STREAM");
                    break;
                }
            }
        }
        //釋放MediaDecoder資源
        mDecoder.stop();
        mDecoder.release();
        mDecoder = null;

        //釋放MediaExtractor資源
        mExtractor.release();
        mExtractor = null;

        //釋放AudioTrack資源
        audioTrack.stop();
        audioTrack.release();
        audioTrack = null;
    }

    public void stop() {
        eosReceived = true;
    }
}

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章