b、将新视频流与原视频流进行帧号同步,使该两个视频流中的各个相同内容的帧一一对应;其中,新视频流是指新编码器实例输出的视频流,原视频流是指原编码器实例输出的视频流;
c、在新视频流中选择一个关键帧,并从该关键帧开始向用户终端传送新视频流,原视频流传送完该关键帧的上一帧数据后结束,且该关键帧的帧号与原视频流的最后一个关键帧的帧号之间的间距大于新视频流GOP长度的1/2;
d、关闭原编码器实例,空出编码器硬件资源,准备下一次切换。
举例
以方式3为例,我们来做个简单Demo实验实现无缝切码流
首先看下切换效果图(两个视频一个是在打球,一个是录风景):
package com.example.hejunlin.seamlessvideo;
import android.annotation.TargetApi;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.pm.PackageManager;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnPreparedListener;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.Gravity;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.widget.FrameLayout;
import static android.Manifest.permission.READ_EXTERNAL_STORAGE;
import static android.Manifest.permission.WRITE_EXTERNAL_STORAGE;
/**
- 兼容8.0版本
*/
public class MainActivity extends AppCompatActivity {
private SurfaceView mVideoSurface, mNextSurface;
private FrameLayout mFrame;
private MediaPlayer mCurrentMediaPlayer, mNextMediaPlayer;
private Handler mHandler;
private int mIndex = 0;
private String path1 = Environment.getExternalStorageDirectory().getAbsolutePath() + “/1.mp4”;
private String path2 = Environment.getExternalStorageDirectory().getAbsolutePath() + “/2.mp4”;
private String[] paths = new String[]{path1, path2};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (!checkPermission()) {
startAction();
} else {
if (checkPermission()) {
requestPermissionAndContinue();
} else {
startAction();
}
}
}
@Override
protected void onDestroy() {
super.onDestroy();
mHandler.removeCallbacks(mPlayRun);
if (mCurrentMediaPlayer != null) {
mCurrentMediaPlayer.release();
mCurrentMediaPlayer = null;
}
if (mNextMediaPlayer != null) {
mNextMediaPlayer.release();
mNextMediaPlayer = null;
}
}
Runnable mPlayRun = new Runnable() {
@Override
public void run() {
Log.d(MainActivity.class.getSimpleName(), "run: ");
mCurrentMediaPlayer.pause();
mNextMediaPlayer.pause();
mNextMediaPlayer.reset();
try {
if (mIndex == 0) {
String path = paths[mIndex % paths.length];
Log.d(MainActivity.class.getSimpleName(), "path1: " + path);
mIndex++;
mCurrentMediaPlayer.setDataSource(MainActivity.this, Uri.parse(path));
mCurrentMediaPlayer.setOnPreparedListener(new OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer player) {
Log.d(MainActivity.class.getSimpleName(), “start 1”);
mCurrentMediaPlayer.start();
mVideoSurface.setVisibility(View.GONE);
}
});
mCurrentMediaPlayer.prepareAsync();
mNextMediaPlayer.setDataSource(MainActivity.this, Uri.parse(path));
mNextMediaPlayer.setOnPreparedListener(new OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer arg0) {
mNextMediaPlayer.start();
}
});
mNextMediaPlayer.prepareAsync();
} else {
String path = paths[mIndex % paths.length];
mIndex++;
Log.d(MainActivity.class.getSimpleName(), "path2: " + path);
mNextMediaPlayer.setDataSource(MainActivity.this, Uri.parse(path));
mNextMediaPlayer.setOnPreparedListener(new OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer arg0) {
mNextMediaPlayer.start();
Log.d(MainActivity.class.getSimpleName(), “start 2”);
}
});
mNextMediaPlayer.prepareAsync();
}
} catch (Exception e) {
e.printStackTrace();
}
mHandler.postDelayed(mPlayRun, 10000); // 第一个视频10s钟
}
};
class VideoSurfaceHodlerCallback implements SurfaceHolder.Callback {
@Override
public void surfaceChanged(
SurfaceHolder holder,
int format, int width, int height) {
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
mCurrentMediaPlayer.setDisplay(mVideoSurface.getHolder());
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
}
}
class NextVideoSurfaceHodlerCallback implements SurfaceHolder.Callback {
@Override
public void surfaceChanged(
SurfaceHolder holder,
int format, int width, int height) {
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
mNextMediaPlayer.setDisplay(mNextSurface.getHolder());
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
}
}
private static final int PERMISSION_REQUEST_CODE = 200;
private boolean checkPermission() {
return ContextCompat.checkSelfPermission(this, WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED
&& ContextCompat.checkSelfPermission(this, READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED
;
}
private void requestPermissionAndContinue() {
if (ContextCompat.checkSelfPermission(this, WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED
&& ContextCompat.checkSelfPermission(this, READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
if (ActivityCompat.shouldShowRequestPermissionRationale(this, WRITE_EXTERNAL_STORAGE)
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
![](https://i-blog.csdnimg.cn/blog_migrate/badaf534b4c5c487000ea2e0354e9776.jpeg)
实战系列
话不多说,Android实战系列集合都已经系统分类好,由于文章篇幅问题没法过多展示
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!**
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
![](https://i-blog.csdnimg.cn/blog_migrate/badaf534b4c5c487000ea2e0354e9776.jpeg)
实战系列
话不多说,Android实战系列集合都已经系统分类好,由于文章篇幅问题没法过多展示
[外链图片转存中…(img-XJ98nbN5-1712376494071)]
[外链图片转存中…(img-ekfnKbWv-1712376494072)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!