`

使用MediaCodec解码使用SurfaceView显示视频

 
阅读更多
From https://www.cnblogs.com/CoderTian/p/6221944.html


1.MainActivity.java

import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.view.SurfaceHolder;
import android.view.SurfaceView;

/**
* MediaCodec SurfaceHolder Example
* @author taehwan
*
*/
public class MainActivity extends Activity implements SurfaceHolder.Callback {
    private VideoDecoderThread mVideoDecoder;

    private static final String FILE_PATH = Environment.getExternalStorageDirectory() + "/TopGirl.mp4";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        SurfaceView surfaceView = new SurfaceView(this);
        surfaceView.getHolder().addCallback(this);
        setContentView(surfaceView);

        mVideoDecoder = new VideoDecoderThread();
    }

    @Override
    public void surfaceCreated(SurfaceHolder holder) {

    }

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width,    int height) {
        if (mVideoDecoder != null) {
            if (mVideoDecoder.init(holder.getSurface(), FILE_PATH)) {
                mVideoDecoder.start();

            } else {
                mVideoDecoder = null;
            }

        }
    }

    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
        if (mVideoDecoder != null) {
            mVideoDecoder.close();
        }
    }

}

2.VideoDecoderThread.java

import java.io.IOException;
import java.nio.ByteBuffer;

import android.media.MediaCodec;
import android.media.MediaCodec.BufferInfo;
import android.media.MediaExtractor;
import android.media.MediaFormat;
import android.util.Log;
import android.view.Surface;

public class VideoDecoderThread extends Thread {
    private static final String VIDEO = "video/";
    private static final String TAG = "VideoDecoder";
    private MediaExtractor mExtractor;
    private MediaCodec mDecoder;

    private boolean eosReceived;

    public boolean init(Surface surface, String filePath) {
        eosReceived = false;
        try {
            mExtractor = new MediaExtractor();
            mExtractor.setDataSource(filePath);
            //分离出音轨和视轨
            Log.d(TAG, "getTrackCount: " + mExtractor.getTrackCount() );
            for (int i = 0; i < mExtractor.getTrackCount(); i++) {
                MediaFormat format = mExtractor.getTrackFormat(i);

                String mime = format.getString(MediaFormat.KEY_MIME);
                if (mime.startsWith(VIDEO)) {
                    mExtractor.selectTrack(i);
                    mDecoder = MediaCodec.createDecoderByType(mime);
                    try {
                        Log.d(TAG, "format : " + format);
                        mDecoder.configure(format, surface, null, 0 /* Decoder */);

                    } catch (IllegalStateException e) {
                        Log.e(TAG, "codec '" + mime + "' failed configuration. " + e);
                        return false;
                    }

                    mDecoder.start();
                    break;
                }
            }

        } catch (IOException e) {
            e.printStackTrace();
        }

        return true;
    }

    @Override
    public void run() {
        BufferInfo info = new BufferInfo();
        ByteBuffer[] inputBuffers = mDecoder.getInputBuffers();
        mDecoder.getOutputBuffers();

        boolean isInput = true;
        boolean first = false;
        long startWhen = 0;

        while (!eosReceived) {
            if (isInput) {
                int inputIndex = mDecoder.dequeueInputBuffer(10000);
                if (inputIndex >= 0) {
                    // fill inputBuffers[inputBufferIndex] with valid data
                    ByteBuffer inputBuffer = inputBuffers[inputIndex];

                    int sampleSize = mExtractor.readSampleData(inputBuffer, 0);

                    if (mExtractor.advance() && sampleSize > 0) {
                        mDecoder.queueInputBuffer(inputIndex, 0, sampleSize, mExtractor.getSampleTime(), 0);

                    } else {
                        Log.d(TAG, "InputBuffer BUFFER_FLAG_END_OF_STREAM");
                        mDecoder.queueInputBuffer(inputIndex, 0, 0, 0, MediaCodec.BUFFER_FLAG_END_OF_STREAM);
                        isInput = false;
                    }
                }
            }

            int outIndex = mDecoder.dequeueOutputBuffer(info, 10000);
            switch (outIndex) {
                case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED:
                    Log.d(TAG, "INFO_OUTPUT_BUFFERS_CHANGED");
                    mDecoder.getOutputBuffers();
                    break;

                case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED:
                    Log.d(TAG, "INFO_OUTPUT_FORMAT_CHANGED format : " + mDecoder.getOutputFormat());
                    break;

                case MediaCodec.INFO_TRY_AGAIN_LATER:
//                Log.d(TAG, "INFO_TRY_AGAIN_LATER");
                    break;

                default:
                    if (!first) {
                        startWhen = System.currentTimeMillis();
                        first = true;
                    }
                    try {
                        long sleepTime = (info.presentationTimeUs / 1000) - (System.currentTimeMillis() - startWhen);
                        //Log.d(TAG, "info.presentationTimeUs : " + (info.presentationTimeUs / 1000) + " playTime: " + (System.currentTimeMillis() - startWhen) + " sleepTime : " + sleepTime);

                        if (sleepTime > 0)
                            Thread.sleep(sleepTime);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }

                    mDecoder.releaseOutputBuffer(outIndex, true /* Surface init */);
                    break;
            }

            // All decoded frames have been rendered, we can stop playing now
            if ((info.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0) {
                Log.d(TAG, "OutputBuffer BUFFER_FLAG_END_OF_STREAM");
                break;
            }
        }

        mDecoder.stop();
        mDecoder.release();
        mExtractor.release();
    }

    public void close() {
        eosReceived = true;
    }
}
分享到:
评论

相关推荐

    Android 硬解码MediaCodecDemo

    2. 设置输入Surface:为了显示解码后的视频帧,我们需要提供一个Surface,通常是通过SurfaceView或TextureView创建的。调用`MediaCodec.setOutputSurface()`将Surface传递给MediaCodec。 3. 提供输入缓冲区:获取...

    Android使用MediaCodec将摄像头采集的视频编码为h264

    首先,我们需要创建一个SurfaceView来预览摄像头采集的视频,然后使用Camera API来获取摄像头采集的视频流数据。接着,我们使用MediaCodec将视频流数据编码为h264,并将其保存到文件中。 在 MediaCodec中,我们可以...

    Android MediaCodec 硬解码H264

    在Android平台上,MediaCodec是一个非常重要的组件,它提供...在实际项目中,结合`MediaExtractor`用于从媒体文件中提取数据,以及`SurfaceView`或`TextureView`显示视频画面,可以构建出完整的H264视频播放解决方案。

    MediaCodec解码

    在MediaCodecDemo-master这个项目中,你可以找到一个完整的MediaCodec解码示例。该项目通常包含一个Activity,负责创建和配置MediaCodec,读取H264数据,以及处理输入和输出缓冲区。此外,还可能包含一个SurfaceView...

    MediaCodec编解码

    摄像头采集视频,通过MediaCodec编解码,SurfaceView显示

    android MediaCodec 实现h264硬编解码全过程

    MediaCodec 实现h264硬编解码全过程,视频数据从摄像头读出 yv12格式,转换为I420,投递给encoder,再从encoder取出编码后的h264数据投递给decoder后显示到surfaceView; 实现了udp将h264数据发送到指定主机,可通过...

    Android MediaCodec

    - SurfaceView是Android用于显示视频和图形的一个视图组件,它有自己的独立渲染线程,适合实时显示视频帧。 - 将解码后的视频帧绑定到SurfaceView的Surface上,通过调用SurfaceHolder的`setCallback()`方法注册回...

    Android利用MediaCodec硬解码H264,AAC文件并播放Demo

    以上是使用MediaCodec进行硬解码的基本流程,实际应用中可能需要处理更多细节,例如处理SPS、PPS信息、处理B帧、处理不同类型的NAL单元、同步音频和视频播放等。同时,考虑到兼容性问题,需要检查设备是否支持硬件...

    MediaCodec-Decode-selectFile-master.zip 安卓Android 解码播放

    总之,"MediaCodec-Decode-selectFile-master"项目提供了一个学习和实践Android MediaCodec解码播放的好例子。通过深入理解和应用这个项目,开发者能够掌握如何利用硬件加速技术在Android上实现高效、流畅的多媒体...

    MediaCodec,使用mediacodec的音频和视频录制演示。.zip

    - **Surface和SurfaceView**:视频录制过程中,常常需要一个Surface来显示预览画面,这通常通过SurfaceView或TextureView实现。MediaCodec可以将编码后的视频帧写入Surface,供上层应用显示。 3. **MediaMuxer集成...

    android mediacodec 硬编解码

    预览Surface通常由Camera2 API提供,而解码后的Surface可以用来显示在TextureView或者SurfaceView上。 ### 5. 三星S6的兼容性 三星Galaxy S6支持硬件编解码功能,因此在这款设备上实现Mediacodec的硬编解码是可行的...

    MediaCodec使用回调方式进行视频解码学习demo

    输出格式通常是Surface,用于显示视频帧。 4. **输入缓冲区**:解码过程开始时,需要将视频流的数据填入到输入缓冲区。这可以通过`MediaCodec.dequeueInputBuffer(int timeoutUs)`方法获取输入缓冲区索引,然后调用...

    MediaCodec 投屏demo

    投屏功能的实现,MediaCodec编解码,MediaProjection获取屏幕视频,WebSocket传视频数据,SurfaceView显示。小demo,演示使用。时间有点赶,重新打包了一份,有问题可以交流一下。

    MediaCodecApp

    android 4.1.2版本以上,利用摄像头像采图像,mediacodec进行H264编码,mediacodec解码并在surfaceview上显示。实现android上H264视频自编自解。实测可用。 注:对于不同的手机,编码器的colorformat有所不同,请有...

    硬编码,使用SurfaceView和Camera取帧并保存本地.h264和上传Socket实时视频

    实现这一过程通常需要使用MediaCodec,它是Android提供的硬件加速编码和解码的API。我们需要配置MediaCodec为编码模式,输入原始图像数据,然后获取编码后的H264 NAL单元,这些单元可以被保存为本地.h264文件,也...

    surfaceview网络视频播放

    在实际项目中,可能会涉及到更复杂的场景,比如使用自定义的MediaExtractor或MediaCodec来处理解码,或者使用ijkplayer、ExoPlayer等第三方库来增强播放功能和兼容性。`tzt_mediavideo`这个文件可能包含了一些示例...

    android mediacodec硬编解码

    本实例主要涉及如何使用MediaCodec进行硬编解码,以及实现图像预览和解码显示的功能,适用于如三星S6等设备。 1. **MediaCodec API简介** MediaCodec是一个接口,它提供了与设备硬件编解码器交互的方法。通过创建...

    Android H264数据解帧和播放.zip

    最后,我们使用SurfaceView显示解码后的视频帧。SurfaceView是一个特殊的视图,提供了与硬件加速的图形渲染器交互的能力,非常适合用于视频播放。我们需要创建一个SurfaceHolder并将其关联到SurfaceView,然后将...

    Android传屏代码

    本资源包包含发送App和接收端APp,均包在Android6.0以上。通过ScreenRecord截屏编码成H264,然后在通过Tcp发送到接受端。接收端通过Mediacodec解码显示在SurfaceView上面。SPS和PPS FrameRate也是通过TCP发送

    Android从RTMP流中提取H264和AAC数据进行播放.zip

    Android从RTMP流中提取H264和AAC数据进行播放 音频播放通过MediaCodec+AudioTrack完成 视频播放通过MediaCodec+SurfaceView完成 RTMP解码通过librtmp完成

Global site tag (gtag.js) - Google Analytics