android surfaceview ffmpeg的渲染简单说明

首先在java层面中,surfaceview的显示是通过SurfaceHolder这个类中2个方法实现的:

Canvas SurfaceHolder.lockCanvas();

SurfaceHolder.unlockAndPost(Canvas canvas);


例如

  1. Canvas canvas = holder.lockCanvas();  
  2. canvas.drawRGB(rand.nextInt(255), rand.nextInt(255), rand.nextInt(255));  
  3. holder.unlockCanvasAndPost(canvas);  

先锁定surface然后返回canvas,然后使用canvas绘制,最后再解锁并渲染到屏幕上可见。


那么在ffmpeg中渲染surfaceview只需要通过一个native方法把surface传入就行。

例如

@Override

public void surfaceCreated(SurfaceHolder holder) {

new Thread(new Runnable() {

@Override

public void run() {

play(surfaceHolder.getSurface());

}

}).start();

}

...


public native int play(Object surface);


然后在需要渲染的线程中

// 获取native window

ANativeWindow *nativeWindow = ANativeWindow_fromSurface(env, surface);


最后在每一帧要渲染处

// lock native window buffer

ANativeWindow_lock(nativeWindow, &windowBuffer, 0);

...

ANativeWindow_unlockAndPost(nativeWindow);



具体参考

http://blog.csdn.net/wangdong20/article/details/8572835

http://blog.csdn.net/King1425/article/details/71514318?locationNum=4&fps=1










很抱歉,Android系统不支持 VAAPI,它是用于 Linux 等操作系统的视频加速 API。在 Android 中,可以使用硬件加速来加速视频播放,例如使用 MediaCodec 和 SurfaceView 进行视频解码和渲染。下面是一个简单的示例代码: ```java import android.media.MediaCodec; import android.media.MediaExtractor; import android.media.MediaFormat; import android.media.MediaCodec.BufferInfo; import android.view.Surface; import android.view.SurfaceHolder; import android.view.SurfaceView; import java.io.IOException; import java.nio.ByteBuffer; public class VideoPlayer implements SurfaceHolder.Callback { private SurfaceView mSurfaceView; private Surface mSurface; private MediaExtractor mExtractor; private MediaCodec mDecoder; private boolean mIsPlaying; public VideoPlayer(SurfaceView surfaceView) { mSurfaceView = surfaceView; mSurfaceView.getHolder().addCallback(this); } public void play(String videoPath) throws IOException { mExtractor = new MediaExtractor(); mExtractor.setDataSource(videoPath); int trackIndex = selectTrack(mExtractor); mExtractor.selectTrack(trackIndex); MediaFormat mediaFormat = mExtractor.getTrackFormat(trackIndex); String mime = mediaFormat.getString(MediaFormat.KEY_MIME); mDecoder = MediaCodec.createDecoderByType(mime); mDecoder.configure(mediaFormat, mSurface, null, 0); mDecoder.start(); ByteBuffer[] inputBuffers = mDecoder.getInputBuffers(); ByteBuffer[] outputBuffers = mDecoder.getOutputBuffers(); BufferInfo bufferInfo = new BufferInfo(); mIsPlaying = true; while (mIsPlaying) { int inputBufferIndex = mDecoder.dequeueInputBuffer(10000); if (inputBufferIndex >= 0) { ByteBuffer inputBuffer = inputBuffers[inputBufferIndex]; int sampleSize = mExtractor.readSampleData(inputBuffer, 0); if (sampleSize < 0) { mDecoder.queueInputBuffer(inputBufferIndex, 0, 0, 0, MediaCodec.BUFFER_FLAG_END_OF_STREAM); mIsPlaying = false; } else { long presentationTimeUs = mExtractor.getSampleTime(); mDecoder.queueInputBuffer(inputBufferIndex, 0, sampleSize, presentationTimeUs, 0); mExtractor.advance(); } } int outputBufferIndex = mDecoder.dequeueOutputBuffer(bufferInfo, 10000); switch (outputBufferIndex) { case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED: outputBuffers = mDecoder.getOutputBuffers(); break; case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED: // do nothing break; case MediaCodec.INFO_TRY_AGAIN_LATER: // do nothing break; default: ByteBuffer outputBuffer = outputBuffers[outputBufferIndex]; mDecoder.releaseOutputBuffer(outputBufferIndex, true); break; } } mDecoder.stop(); mDecoder.release(); mExtractor.release(); } private int selectTrack(MediaExtractor extractor) { int numTracks = extractor.getTrackCount(); for (int i = 0; i < numTracks; i++) { MediaFormat format = extractor.getTrackFormat(i); String mime = format.getString(MediaFormat.KEY_MIME); if (mime.startsWith("video/")) { return i; } } return -1; } @Override public void surfaceCreated(SurfaceHolder holder) { mSurface = holder.getSurface(); } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { // do nothing } @Override public void surfaceDestroyed(SurfaceHolder holder) { mSurface = null; } } ``` 使用示例: ```java VideoPlayer player = new VideoPlayer(surfaceView); try { player.play(videoPath); } catch (IOException e) { e.printStackTrace(); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值