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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值