每次我暂停我的活动(实际上是片段)转到另一个应用程序时,当我带着onResume返回时,我尝试恢复视频播放,但它没有播放:我得到一个空白屏幕。经过调查,我在“逻辑猫”中看到了以下内容
E/BufferQueueProducer: [unnamed-23827-0] queueBuffer: BufferQueue has been abandoned
E/MediaPlayer: error (1, -38)
E/MediaPlayer: error (1, -38)
E/MediaPlayer: error (1, -38)
E/MediaPlayer: error (1, -38)
E/BufferQueueProducer: [unnamed-23827-0] connect(P): BufferQueue has been abandoned
这是我在简历中调用的代码
player.seekTo(mVideoSeekPosition);
player.start();
FYI:我一直在尝试将这个答案应用到我的案例中,但我做不到:What can I do when the BufferQueue has been abandoned?
更新
我挣扎着一个人去,但我仍然在崩溃。所以我把所有的代码都贴出来了
private void setupVideoPlayingSystem(View root) {
textureView = (TextureView) root.findViewById(R.id.textureView);
textureView.setSurfaceTextureListener(this);
}
@Override
public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int width, int height) {
Log.d(TAG, "onSurfaceTextureAvailable");
if (null == surface) {
Log.d(TAG, "new surface");
surface = new Surface(surfaceTexture);
mediaPlayer = new MediaPlayer();
mediaPlayer.setSurface(surface);
mediaPlayer.setLooping(false);
}
/*
outstandingVideoRequest is IOU for orentation change (verifed: onResume before onSurfaceTextureAvailable)
but for cold startup, must check mVideoUrl
*/
if (outstandingVideoRequest && null != mVideoUrl) {
outstandingVideoRequest = false;
playNewVideo(mVideoUrl);
}
}
@Override
public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {
Log.d(TAG, "onSurfaceTextureSizeChanged");
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
Log.d(TAG, "onSurfaceTextureDestroyed");
return false;//leave destruction for onDestroy
}
@Override
public void onSurfaceTextureUpdated(SurfaceTexture surface) {
}
private void playNewVideo(String url) {
if (null == mediaPlayer || null == surface) {
Log.d(TAG, "playNewVideo not ready");
synchronized (outstandingVideoRequest) {
Log.d(TAG, "playNewVideo outstandingVideoRequest");
outstandingVideoRequest = true;
}
} else {
try {
mediaPlayer.reset();
mediaPlayer.setDataSource(getContext(), Uri.parse(url));
mediaPlayer.setLooping(false);
mediaPlayer.prepareAsync();
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer player) {
Log.d(TAG, "onPrepared changeMediaPlayerDatasource");
onReadyToPlay(player);
}
});
} catch (Exception e) {//IOException && IllegalStateException
Log.d(TAG, "textureview playNewVideo ERORR");
e.printStackTrace();
}
}
}
private void resumeVideoUponReturningFromAnotherActivity() {
if (null == mediaPlayer || null == surface) {
Log.d(TAG, "resumeVideoUponReturningFromAnotherActivity outstandingVideoRequest");
outstandingVideoRequest = true;
} else {
// playNewVideo(mVideoUrl);
Log.d(TAG, "resumeVideoUponReturningFromAnotherActivity go NOW");
mediaPlayer.setSurface(surface);
onReadyToPlay(mediaPlayer);
}
}
private void onReadyToPlay(MediaPlayer player) {
//play video
mProgressCircle.setVisibility(View.GONE);
showVideoOverlayChildren();
if (0 == mVideoSeekPosition) {
Log.d(TAG, "onReadyToPlay start");
player.start();
} else {
Log.d(TAG, "onReadyToPlay seek");
player.seekTo(mVideoSeekPosition);
player.start();
}
mHandler.postDelayed(new Runnable() {
@Override
public void run() {
Log.d(TAG, "postDelayed resumeVideo");
hideVideoOverlayChildren();
}
}, Constant.BEFORE_VIDEO_OVERLAY_DISAPPEAR);
}
private void destroyMediaPlayer() {
if (null != mediaPlayer) {//move to video todo
mediaPlayer.stop();
mediaPlayer.release();
mediaPlayer = null;
}
if (null != surface) {
surface.release();
surface = null;
}
}
private void pauseVideo() {
if (null != mediaPlayer) {
Log.d(TAG, "pause");
mediaPlayer.pause();
mVideoSeekPosition = mediaPlayer.getCurrentPosition();
}
}
private void stopVideo(){
if (null != mediaPlayer) {
Log.d(TAG, "stop video");
mediaPlayer.pause();
mVideoSeekPosition = mediaPlayer.getCurrentPosition();
mediaPlayer.stop();
}
}
@Override
public void onResume() {
super.onResume();
Log.d(TAG, "onResume");
mLocalBroadcastManager.registerReceiver(mVideoSelectionReceiver, mVideoSelectedIntentFilter);
resumeVideoUponReturningFromAnotherActivity();
}
发布于 2016-02-25 15:57:13
在活动之间切换时,我也遇到了同样的问题,还有MediaPlayer(1971):Error (100,0)。通过在onSurfaceTextureDestroyed中添加这些行来解决这个问题
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
if (mediaPlayer != null) {
mediaPlayer.stop();
mediaPlayer.release();
mediaPlayer = null;
}
return true;
}
发布于 2017-11-22 09:19:19
我发现setSurface(null)
很有用。
如果使用TextureView显示某些内容,则在调用TextureView.SurfaceTextureListener
回调onSurfaceTextureDestroyed
时,必须停止使用由camera2
、MediaCodec
或MediaPlayer
绑定的SurfaceTexture/new Surface(SurfaceTexture)
。
像这样
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
mediaPlayer.setDisplayer(null);
return false;//do not return true if you reuse it.
}
发布于 2016-10-12 13:22:30
您的代码中似乎有一个bug :在SurfaceTextureDestroyed()中,您没有重置曲面或mediaPlayer。当恢复时,mediaPlayer和surface都不是空的,因此在resumeVideoUponReturningFromAnotherActivity()中设置曲面并调用开始播放,但是由于以前的SurfaceTextureDestroyed,SurfaceTextureDestroyed已经无效。所以你才会犯错。
要修复它,您应该在回调SurfaceTextureDestroyed中重置曲面。恢复时,在回调SurfaceTextureAvailable中重新生成曲面,将其设置为mediaPlayer并调用开始播放。密码是这样的:
public void onResume() {
if (mSurface == null) {
mResumeRequested = true;
return;
}
mMediaPlayer.start();
}
@Override
public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {
mSurface = new Surface(surface);
if (mMediaPlayer != null) {
mMediaPlayer.setSurface(mSurface);
if (mResumeRequested) {
mMediaPlayer.start();
mResumeRequested = false;
}
}
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
mSurface = null;
return false;
}
你根本不需要重置媒体播放器。如果您重置,您必须重新实例化它和重新缓冲,这会导致延迟,这会损害用户体验,因为没有延迟暂停/恢复是更需要的。
https://stackoverflow.com/questions/33288303
复制相似问题