在 4.0.4 中删除包含 videoView 的片段时出错

Error while removing fragment containing videoView in 4.0.4

我有一个包含 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();
    }

每当我在播放视频时删除此片段,IllegalStateException 仅在 Android 4.0.4 中抛出(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.access00(VideoView.java:49)
    at android.widget.VideoView.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.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 的原因是什么?我该如何解决? 任何指向解决方案的东西都会有很大帮助。

我想你在 release() 之后调用了 reset() 并且这会抛出 illegalstateexception。

查看此 URL 上给出的代码。这是来自实际的 android github 回购协议。 https://github.com/android/platform_frameworks_base/blob/866658261f4613e17ed6f39a74975ad0c9f40767/media/jni/android_media_MediaPlayer.cpp

java 中的重置函数调用本机 _reset 函数,该函数在本 URL 的 CPP 文件中给出,这很难调试,因为它是无法调试的 JNI 调用.所以在我看来,除非我们真的知道你到底在做什么,否则很难知道出了什么问题

我建议你换一种方式。在删除片段之前。在尝试删除片段之前调用此代码。

videoView.stopPlayback();
videoView.suspend();

从堆栈跟踪可以明显看出,当您删除片段时,Android 所做的是删除 VideoView(毕竟这是一个视图),这会导致 Videoview正在尝试释放底层 Mediaplayer。但 Mediaplayer 本身可能不是导致此问题的要发布的状态。理想情况下,这口井可能已在 VideocView 中处理,但看起来没有