- 浏览: 216799 次
- 性别:
- 来自: 上海
-
文章分类
- 全部博客 (196)
- android (153)
- sqlite (1)
- java (3)
- os (4)
- tools (4)
- ResolveInfo 转 packageinf 取信息 (1)
- multimedia (2)
- fastboot (1)
- create user in samba (0)
- android qcn (0)
- windows (0)
- net (2)
- linux (1)
- git (0)
- proguard (0)
- android cpu (1)
- android gts (1)
- MySql (1)
- c/c++ (0)
- network (1)
- python (3)
- AI (1)
- Ultroedit (1)
- HarmonyOS NEXT (0)
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;
}
}
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 SnappyDB
2018-05-15 11:23 993https://blog.csdn.net/qq_214305 ... -
使用UncaughtExceptionHandler保存全局异常(转)
2018-05-07 15:04 664From: https://blog.csdn.net/dre ... -
android udp
2018-04-10 19:47 623Android之Socket的基于UDP传输 接收方创建步骤: ... -
android DownloadManager
2018-04-10 12:52 617From: https://www.jianshu.com/p ... -
Snackbar 可以交互的Toast
2018-04-08 21:44 790Snackbar 是 Android design sup ... -
android 操作串口
2018-03-19 18:20 683http://blog.csdn.net/akunainian ... -
proguard对 android-support-v4.jar 的配置
2018-03-16 15:32 743-libraryjars libs/android-sup ... -
apk 多个dex
2018-03-16 10:05 1025android studio 配置应用生成多个dex ... -
ddmlib 解决android n file explorer 为空问题
2018-03-16 09:27 968百度云:http://pan.baidu.com/s/1sl4 ... -
android 代码混淆
2018-03-15 08:38 1076打开混淆开关 Android.mk ... -
menu item 显示icon
2018-03-07 09:32 710@Override public boolean onMenu ... -
ubuntu 脚本切换root用户,并执行其它脚本
2018-03-05 09:58 0#!/bin/bash expect -c" ... -
DrawerLayout 滑动侧边栏
2018-03-01 10:42 0// TODO -
jack-server 多用户编译问题
2018-02-27 10:44 14173),需要同时修改 $HOME/.jack-server/co ... -
更新eclipse 遇到两个问题 Oxygen.2 Release (4.7.2)
2018-02-26 14:28 12471. the file dx.jar was not load ... -
监听系统开关变化
2018-02-05 14:02 820// 转屏开关 /** * Content obse ... -
android 8.0 模块编译命令
2018-02-02 14:42 1184Invoke ". build/envsetup.s ... -
svn checkout 忽略某个文件夹
2018-01-26 11:14 1377android源码更新时, frameworks\compil ... -
eclipse jni
2018-01-19 16:10 500试过,可以运行 http://blog.csdn.net/sb ... -
opencv for android
2018-01-19 14:37 455http://blog.csdn.net/sbsujjbcy/ ...
相关推荐
2. 设置输入Surface:为了显示解码后的视频帧,我们需要提供一个Surface,通常是通过SurfaceView或TextureView创建的。调用`MediaCodec.setOutputSurface()`将Surface传递给MediaCodec。 3. 提供输入缓冲区:获取...
首先,我们需要创建一个SurfaceView来预览摄像头采集的视频,然后使用Camera API来获取摄像头采集的视频流数据。接着,我们使用MediaCodec将视频流数据编码为h264,并将其保存到文件中。 在 MediaCodec中,我们可以...
在Android平台上,MediaCodec是一个非常重要的组件,它提供...在实际项目中,结合`MediaExtractor`用于从媒体文件中提取数据,以及`SurfaceView`或`TextureView`显示视频画面,可以构建出完整的H264视频播放解决方案。
在MediaCodecDemo-master这个项目中,你可以找到一个完整的MediaCodec解码示例。该项目通常包含一个Activity,负责创建和配置MediaCodec,读取H264数据,以及处理输入和输出缓冲区。此外,还可能包含一个SurfaceView...
摄像头采集视频,通过MediaCodec编解码,SurfaceView显示
MediaCodec 实现h264硬编解码全过程,视频数据从摄像头读出 yv12格式,转换为I420,投递给encoder,再从encoder取出编码后的h264数据投递给decoder后显示到surfaceView; 实现了udp将h264数据发送到指定主机,可通过...
- SurfaceView是Android用于显示视频和图形的一个视图组件,它有自己的独立渲染线程,适合实时显示视频帧。 - 将解码后的视频帧绑定到SurfaceView的Surface上,通过调用SurfaceHolder的`setCallback()`方法注册回...
以上是使用MediaCodec进行硬解码的基本流程,实际应用中可能需要处理更多细节,例如处理SPS、PPS信息、处理B帧、处理不同类型的NAL单元、同步音频和视频播放等。同时,考虑到兼容性问题,需要检查设备是否支持硬件...
总之,"MediaCodec-Decode-selectFile-master"项目提供了一个学习和实践Android MediaCodec解码播放的好例子。通过深入理解和应用这个项目,开发者能够掌握如何利用硬件加速技术在Android上实现高效、流畅的多媒体...
- **Surface和SurfaceView**:视频录制过程中,常常需要一个Surface来显示预览画面,这通常通过SurfaceView或TextureView实现。MediaCodec可以将编码后的视频帧写入Surface,供上层应用显示。 3. **MediaMuxer集成...
预览Surface通常由Camera2 API提供,而解码后的Surface可以用来显示在TextureView或者SurfaceView上。 ### 5. 三星S6的兼容性 三星Galaxy S6支持硬件编解码功能,因此在这款设备上实现Mediacodec的硬编解码是可行的...
输出格式通常是Surface,用于显示视频帧。 4. **输入缓冲区**:解码过程开始时,需要将视频流的数据填入到输入缓冲区。这可以通过`MediaCodec.dequeueInputBuffer(int timeoutUs)`方法获取输入缓冲区索引,然后调用...
投屏功能的实现,MediaCodec编解码,MediaProjection获取屏幕视频,WebSocket传视频数据,SurfaceView显示。小demo,演示使用。时间有点赶,重新打包了一份,有问题可以交流一下。
android 4.1.2版本以上,利用摄像头像采图像,mediacodec进行H264编码,mediacodec解码并在surfaceview上显示。实现android上H264视频自编自解。实测可用。 注:对于不同的手机,编码器的colorformat有所不同,请有...
实现这一过程通常需要使用MediaCodec,它是Android提供的硬件加速编码和解码的API。我们需要配置MediaCodec为编码模式,输入原始图像数据,然后获取编码后的H264 NAL单元,这些单元可以被保存为本地.h264文件,也...
在实际项目中,可能会涉及到更复杂的场景,比如使用自定义的MediaExtractor或MediaCodec来处理解码,或者使用ijkplayer、ExoPlayer等第三方库来增强播放功能和兼容性。`tzt_mediavideo`这个文件可能包含了一些示例...
本实例主要涉及如何使用MediaCodec进行硬编解码,以及实现图像预览和解码显示的功能,适用于如三星S6等设备。 1. **MediaCodec API简介** MediaCodec是一个接口,它提供了与设备硬件编解码器交互的方法。通过创建...
最后,我们使用SurfaceView显示解码后的视频帧。SurfaceView是一个特殊的视图,提供了与硬件加速的图形渲染器交互的能力,非常适合用于视频播放。我们需要创建一个SurfaceHolder并将其关联到SurfaceView,然后将...
本资源包包含发送App和接收端APp,均包在Android6.0以上。通过ScreenRecord截屏编码成H264,然后在通过Tcp发送到接受端。接收端通过Mediacodec解码显示在SurfaceView上面。SPS和PPS FrameRate也是通过TCP发送
Android从RTMP流中提取H264和AAC数据进行播放 音频播放通过MediaCodec+AudioTrack完成 视频播放通过MediaCodec+SurfaceView完成 RTMP解码通过librtmp完成