在 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 执行放大/缩小?

Videoview 在框架布局内没有全屏显示

Acer Iconia A200 上的 VideoView getCurrentPosition() 异常

WebView 之上的 VideoView