在 4.0.4 中删除包含 videoView 的片段时出错
Posted
技术标签:
【中文标题】在 4.0.4 中删除包含 videoView 的片段时出错【英文标题】:Error while removing fragment containing videoView in 4.0.4 【发布时间】:2017-01-17 14:32:18 【问题描述】:我有一个包含 VideoView 和一些其他视图的片段。
public class PlayerPane extends Fragment
... // static variables
private ImageView imageView;
private ImageView gifView;
private VideoView videoView;
private WebView webView;
private PDFView pdfView;
private MyScrollTextView scrollTextView;
private MediaPlayer audioPlayer;
...
@Override
public void onDestroyView()
if (videoView != null && videoView.isPlaying())
LOGGER.info("Stopping videoView");
videoView.stopPlayback();
videoView.suspend();
videoView = null;
super.onDestroyView();
每当我在播放视频时删除此片段时,只会在 android 4.0.4 中引发 IllegalStateException(4.0.4 以上版本没有问题)。
以下是例外:
java.lang.IllegalStateException
at android.media.MediaPlayer._reset(Native Method)
at android.media.MediaPlayer.reset(MediaPlayer.java:1236)
at android.widget.VideoView.release(VideoView.java:549)
at android.widget.VideoView.access$2300(VideoView.java:49)
at android.widget.VideoView$6.surfaceDestroyed(VideoView.java:537)
at android.view.SurfaceView.updateWindow(SurfaceView.java:581)
at android.view.SurfaceView.onWindowVisibilityChanged(SurfaceView.java:290)
at android.view.View.dispatchDetachedFromWindow(View.java:9823)
at android.view.ViewGroup.dispatchDetachedFromWindow(ViewGroup.java:2266)
at android.view.ViewGroup.removeViewInternal(ViewGroup.java:3588)
at android.view.ViewGroup.removeViewInternal(ViewGroup.java:3568)
at android.view.ViewGroup.removeView(ViewGroup.java:3516)
at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:951)
at android.app.FragmentManagerImpl.removeFragment(FragmentManager.java:1123)
at android.app.BackStackRecord.run(BackStackRecord.java:592)
at android.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1382)
at android.app.FragmentManagerImpl$1.run(FragmentManager.java:426)
at android.os.Handler.handleCallback(Handler.java:605)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:4424)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:511)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
这就是我删除片段的方式:
getFragmentManager.beginTransaction().remove(fragmentToBeRemoved).commit();
是什么导致了这个 IllegalStateException,我该如何解决这个问题? 任何指向解决方案的内容都会有很大帮助。
【问题讨论】:
为什么要删除片段,我的意思是您可以简单地替换为其他片段 您应该提供minimal reproducible example 以希望找到您问题的具体答案。在这里,我们只有破坏部分。 【参考方案1】:我建议您采取不同的方法。在删除片段之前。在尝试移除片段之前调用此代码。
videoView.stopPlayback();
videoView.suspend();
从堆栈跟踪中可以明显看出,当您删除片段时,Android 所做的是删除 VideoView(毕竟它是一个视图),这导致 Videoview 试图释放底层的 Mediaplayer。但是 Mediaplayer 本身可能不是导致此问题的要发布的状态。理想情况下,这口井可能已在 VideocView 中处理,但看起来没有
【讨论】:
我已经尝试过 stopPlayback 和暂停,但它们没有帮助。这些在 Fragment 的 onDestroyView 中调用。您是否建议我在 onDestroyView 之外调用这些方法? 是的,我建议您在 destroView 之外调用它们。在你这样做之前打电话给他们getFragmentManager.beginTransaction().remove(fragmentToBeRemoved).commit();
不同之处在于,你不是在清理资源,而是按顺序明确地执行它们。无论如何,健壮的系统都是为处理这种情况而设计的,但在这种情况下却不能。试一试【参考方案2】:
我认为你在 release() 之后调用 reset() 会抛出非法状态异常。
【讨论】:
我很确定我没有明确地这样做。另外,我只在 4.0.4 中遇到上述问题,而不是以上任何版本。【参考方案3】:查看此 URL 上给出的代码。这是来自实际的 android github repo。 https://github.com/android/platform_frameworks_base/blob/866658261f4613e17ed6f39a74975ad0c9f40767/media/jni/android_media_MediaPlayer.cpp
来自 java 的 reset 函数调用了本机 _reset 函数,该函数在此 URL 的 CPP 文件中给出,这很难调试,因为它是无法调试的 JNI 调用。所以在我看来,除非我们真的知道你在做什么,否则很难知道出了什么问题
【讨论】:
我已经添加了关于我正在做什么的精确细节。上述问题仅发生在 4.0.4 上,而不是以上任何版本。未测试以下版本。但是,如果需要添加更多细节,请告诉我,我们很乐意这样做。我的首要任务是解决问题。以上是关于在 4.0.4 中删除包含 videoView 的片段时出错的主要内容,如果未能解决你的问题,请参考以下文章
Android VideoView:从父视图中删除视频后变黑
即使 VideoView 在不同的 xml 中(使用包含),全屏视频也是如此
如何在 android 中对 VideoView 执行放大/缩小?