YouTube Android Player API 在解组时抛出“BadParcelableException ClassNotFoundException:asc”与新的 YouTube 版本

Posted

技术标签:

【中文标题】YouTube Android Player API 在解组时抛出“BadParcelableException ClassNotFoundException:asc”与新的 YouTube 版本【英文标题】:YouTube Android Player API throws "BadParcelableException ClassNotFoundException when unmarshalling: asc" with new YouTube version 【发布时间】:2017-11-06 20:45:18 【问题描述】:

向 YouTube Android Player API 库工程师提交错误:请参阅 android-youtube-api 标签

在过去一周半的时间里,我注意到这个奇怪的 BadParcelableException 在我们的应用程序中稳步增加,并将其归结为 YouTube 在 android 上的新版本。

如果您的应用正在播放 youtube 视频、将您的应用置于后台、强制停止 Youtube 应用并再次恢复您的应用,则会发生此崩溃。在 Youtube 版本 12.19.56 上可重现崩溃。还在较旧的 YouTube 版本 12.05.21 上进行了测试,并且没有出现崩溃。

堆栈跟踪:

main Exception: Unable to start activity ComponentInfocom.myapp.MainActivity: 
android.os.BadParcelableException: ClassNotFoundException when unmarshalling: asc 
Stack: java.lang.RuntimeException: Unable to start activity ComponentInfocom.myapp.MainActivity: 
android.os.BadParcelableException: ClassNotFoundException when unmarshalling: asc 
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2666) 
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2727) 
at android.app.ActivityThread.-wrap12(ActivityThread.java) 
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1478) 
at android.os.Handler.dispatchMessage(Handler.java:102) 
at android.os.Looper.loop(Looper.java:154) 
at android.app.ActivityThread.main(ActivityThread.java:6121) 
at java.lang.reflect.Method.invoke(Native Method) 
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:889) 
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:779) Caused by: 
android.os.BadParcelableException: ClassNotFoundException when unmarshalling: asc 
at android.os.Parcel.readParcelableCreator(Parcel.java:2536) 
at android.os.Parcel.readParcelable(Parcel.java:2462) 
at android.os.Parcel.readValue(Parcel.java:2365) 
at android.os.Parcel.readSparseArrayInternal(Parcel.java:2813) 
at android.os.Parcel.readSparseArray(Parcel.java:2068) 
at android.os.Parcel.readValue(Parcel.java:2422) 
at android.os.Parcel.readArrayMapInternal(Parcel.java:2732) 
at android.os.BaseBundle.unparcel(BaseBundle.java:269) 
at android.os.Bundle.getSparseParcelableArray(Bundle.java:934) 
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1208) 
at android.support.v4.app.FragmentManagerImpl.moveFragmentToExpectedState(FragmentManager.java:1528) 
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1595) 
at android.support.v4.app.FragmentManagerImpl.dispatchCreate(FragmentManager.java:2893) 
at android.support.v4.app.FragmentController.dispatchCreate(FragmentController.java:190) 
at android.support.v4.app.FragmentActivity.onCreate(FragmentActivity.java:353) 
at android.support.v7.app.AppCompatActivity.onCreate(AppCompatActivity.java:85) 
at com.myapp.BaseActivity.onCreate(BaseActivity.java:36) 
at com.myapp.MainActivity.onCreate(MainActivity.java:190) 
at android.app.Activity.performCreate(Activity.java:6682) 
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1118) 
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2619) ... 9 more

Caused by: java.lang.ClassNotFoundException: Didn't find class "asc" on path: DexPathList[[zip file "/data/app/com.myapp-naA-_cCrz-w81rqx98ipcQ==/base.apk"],nativeLibraryDirectories=[/data/app/com.myapp-naA-_cCrz-w81rqx98ipcQ==/lib/arm64, /system/lib64, /vendor/lib64]]
at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:93)
at java.lang.ClassLoader.loadClass(ClassLoader.java:379)
at java.lang.ClassLoader.loadClass(ClassLoader.java:312)
... 32 more

更多信息:

@Override
public void onCreate(Bundle savedInstanceState) 
    ...
    super.onCreate(savedInstanceState); // Crashing here MainActivity.java:190
    ...

【问题讨论】:

基于堆栈跟踪,似乎 FragmentManager 在初始化期间试图从 Bundle 中获取它的一些内部状态,Bundle 试图解包自己,但失败了,因为 Bundle 内容以一种微妙的方式被破坏.您能否在通话结束时将 Activity 状态 Bundle 的内容发布到Activity#onSaveInstanceState?覆盖该方法,在返回之前设置断点,然后从调试器复制 整个 Bundle 内容(包括所有嵌套对象及其内容!)并将其粘贴到某个地方供我们查看(Github Gist 等)。特别注意 SparseArrays。 请注意,除了 Bundle 中的错误之外,您自己的代码在这里可能有些内疚。确保您没有因在自己的 Parcelable 中出错而以某种方式损坏 Parcel 数据。 @user1643723 我试过了,但是你如何从调试器中复制一些东西来以文本格式列出它的所有内容? 也许看看 IntelliJ 自定义渲染器?反正也不应该有那么多的信息,就像我说的,看SparseArrays的内容就行了。 您也可以在 onSaveInstanceState 期间使用 LogCat 递归地记录 Bundle 中的所有内容(只是不要将其记录在单行中以确保它适合)。 【参考方案1】:

视图状态保存和恢复的问题。 Youtube 应用程序有一个错误和存储视图状态,其中包含来自 youtube apk 的类 asc 的实例。所以我们的应用程序无法恢复它,因为对这个类一无所知。我的解决方案是通过下一个代码防止在YouTubePlayerSupportFragment 中为YoutubePlayerView 保存视图状态:

@Override
public void onSaveInstanceState(Bundle bundle) 
    super.onSaveInstanceState(bundle);

    // disable view state saving to prevent saving states from youtube apk which cannot be restored.
    View view = getView();
    if (view instanceof ViewGroup) 
        ViewGroup viewGroup = ((ViewGroup) view);
        for (int i = 0; i < viewGroup.getChildCount(); i++) 
            viewGroup.getChildAt(i).setSaveFromParentEnabled(false);
        
    

此代码应添加到YouTubePlayerSupportFragment 的子类中。 此解决方案不会从捆绑包中删除 youtube 播放器状态。这样youtube播放器就恢复成功了。

【讨论】:

在这里尝试了很多不同的解决方案之后,这就是它的工作原理。谢谢! @xkor : getView() 是做什么的? @Saggy 是Fragment 类获取根视图的方法) @xkor : 如何在 Activity 中使用它? @Saggy 你可以在活动布局中找到YoutubePlayerView 并为此视图执行setSaveFromParentEnabled(false);【参考方案2】:

在您的 onCreate 中使用以下代码:

@Override
public void onCreate(Bundle b) 
  if (b != null) 
    final ClassLoader contextCl = getClassLoader();
    b.setClassLoader(contextCl);
  
  super.onCreate(b);

该错误可能是由于 YouTube 库代码将自定义 Parcelable 存储在状态 Bundle 中(YouTube 库类可能被 Proguard 混淆,因此有奇怪的名称,例如“asc”)。

如果上面的建议不起作用,请按照@Fitz 的建议删除状态包。不要试图在 onCreate 中弄乱它(这不起作用),相反,如果您的应用在播放过程中被暂停,最好覆盖 onSaveInstanceState 以返回 Bundle.EMPTY。


请注意,有问题的错误在多个方面都非常棘手。如果您(或 YouTube 应用程序)将嵌套的 Bundles 存储在彼此内部,那么它们还需要设置适当的 ClassLoader……有人应该告诉 Google 获得一些常识,并在 Bundle 中使用 Thread#getContextClassLoader 来一劳永逸地解决这个问题。

【讨论】:

我在我的 MainActivity 上尝试了这个,但没有运气,仍然遇到同样的崩溃。我有 MainActivity,然后是一个带有 ViewPager 的片段,而 ViewPager 的其中一项是包含 YouTubePlayerSupportFragment 的片段【参考方案3】:

我找到了适合我的解决方案。在我的 onCreate 方法中,我捕获了由于 youtube 播放器中的 ClassNotFoundException 而被抛出的 RuntimeException。视频失去其状态。按下播放使其重新开始。否则一切照旧。

【讨论】:

【参考方案4】:

我刚刚使用了一个对我有用的 try catch 块。

try 
    super.onCreate(savedInstanceState);

 catch (BadParcelableException e) 
    // Maybe put an analytic here to see if it is still being thrown.
    // So when youtube fixes the issue in the future we can remove this 
    // ugly try catch.
    Log.e(TAG, e.toString);

【讨论】:

以上是关于YouTube Android Player API 在解组时抛出“BadParcelableException ClassNotFoundException:asc”与新的 YouTube 版本的主要内容,如果未能解决你的问题,请参考以下文章

膨胀类 com.google.android.youtube.player.YouTubePlayerView 时出错

com.google.android.youtube.player.YouTubePlayerSupportFragment.onStart 处的 NullPointerException(未知来源)

无法使用 YouTube Android Player API 播放某些 Youtube 视频

Activity 泄露了原本绑定在这里的 ServiceConnection com.google.android.youtube.player

Activity 泄露了原本绑定在这里的 ServiceConnection com.google.android.youtube.player

YouTube Android Player API 在解组时抛出“BadParcelableException ClassNotFoundException:asc”与新的 YouTube 版本