当解组时,YouTube Android Player API会抛出" BadParcelableException ClassNotFoundException:asc"使用新的YouTube版本

时间:2017-06-06 00:42:52

标签: android youtube parcelable android-youtube-api badparcelableexception

向YouTube Android播放器API库工程师提交错误:请参阅android-youtube-api标记

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

如果您的应用正在播放YouTube视频,将您的应用带到后台,强制停止Youtube应用并再次恢复您的应用,则会发生此崩溃。在Youtube版本12.19.56上可以重现崩溃。还在较旧的YouTube版本12.05.21上进行了测试,但崩溃并不存在。

堆栈追踪:

main Exception: Unable to start activity ComponentInfo{com.myapp.MainActivity}: 
android.os.BadParcelableException: ClassNotFoundException when unmarshalling: asc 
Stack: java.lang.RuntimeException: Unable to start activity ComponentInfo{com.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
    ...
}

5 个答案:

答案 0 :(得分:4)

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

@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的子类中。 此解决方案不会从bundle中删除youtube播放器状态。所以youtube播放器将成功恢复。

答案 1 :(得分:0)

显然这是YouTube应用的错误。 我建议你使用这个hack擦除从youtube客户端传递的包来逃避这个解析错误。

@Override
public void onCreate(Bundle b) {
    if (b != null) {
        b.remove("android:fragments");
    }
    super.onCreate(b);
}

答案 2 :(得分:0)

在onCreate中使用以下代码:

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

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

如果上述建议不起作用,请按照@ Fitz的建议行,并删除状态Bundle。不要试图在onCreate中乱用它(这将无法工作),而是最好覆盖onSaveInstanceState以返回Bundle.EMPTY,如果你的应用程序在播放过程中被暂停。

请注意,有问题的bug在很多方面都非常棘手。如果你(或YouTube应用程序)将嵌套的Bundle存储在彼此之内,那些也需要设置正确的ClassLoader ......有人应该告诉Google获得一些常识并且只需在Bundle中使用Thread#getContextClassLoader来一劳永逸地解决这个问题。 / p>

答案 3 :(得分:0)

我找到了一个适合我的解决方案。在我的onCreate方法中,我抓住因youtube播放器中的RuntimeException而引发的ClassNotFoundException。视频失去了它的状态。按下播放会使其重新开始。否则照常营业。

答案 4 :(得分:-1)

我最后使用了一个适用于我的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);
}