YouTube Android Player API 使用新的 YouTube 版本抛出 "BadParcelableException ClassNotFoundException when unmarshalling: asc"
YouTube Android Player API throws "BadParcelableException ClassNotFoundException when unmarshalling: asc" with new YouTube version
向 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
...
}
在您的 onCreate 中使用以下代码:
@Override
public void onCreate(Bundle b) {
if (b != null) {
final ClassLoader contextCl = getClassLoader();
b.setClassLoader(contextCl);
}
super.onCreate(b);
}
该错误可能是由于 YouTube 库代码将自定义 Parcelable 存储在状态包中(YouTube 库 类 可能被 Proguard 混淆,因此出现了奇怪的名称,如 "asc")。
如果上述建议不起作用,请按照@Fitz 的建议删除状态包。不要试图在 onCreate 中弄乱它(那是行不通的),相反,如果您的应用在播放过程中被暂停,最好将 onSaveInstanceState 重写为 return Bundle.EMPTY。
请注意,所讨论的错误在多个方面都非常棘手。如果您(或 YouTube 应用程序)将嵌套的 Bundle 存储在彼此内部,则它们还需要设置适当的 ClassLoader ......有人应该告诉 Google 获得一些常识,只需在 Bundle 中使用 Thread#getContextClassLoader 一次解决该问题对于所有人。
视图状态保存和恢复的问题。 Youtube 应用程序有一个错误和存储视图状态,其中包含来自 youtube apk 的 class asc
实例。所以我们的应用程序无法恢复它,因为对此一无所知class。我的解决方案是通过下一个代码阻止 YoutubePlayerView
在 YouTubePlayerSupportFragment
中保存视图状态:
@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
的子 class 中。
此解决方案不会从捆绑包中删除 youtube 播放器状态。这样youtube播放器就恢复成功了。
我找到了适合我的解决方案。在我的 onCreate
方法中,我捕获了由于 youtube 播放器中的 ClassNotFoundException
而抛出的 RuntimeException
。视频失去了它的状态。按播放键会重新开始。否则照常营业。
我刚刚使用了一个适合我的 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 播放器 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
...
}
在您的 onCreate 中使用以下代码:
@Override
public void onCreate(Bundle b) {
if (b != null) {
final ClassLoader contextCl = getClassLoader();
b.setClassLoader(contextCl);
}
super.onCreate(b);
}
该错误可能是由于 YouTube 库代码将自定义 Parcelable 存储在状态包中(YouTube 库 类 可能被 Proguard 混淆,因此出现了奇怪的名称,如 "asc")。
如果上述建议不起作用,请按照@Fitz 的建议删除状态包。不要试图在 onCreate 中弄乱它(那是行不通的),相反,如果您的应用在播放过程中被暂停,最好将 onSaveInstanceState 重写为 return Bundle.EMPTY。
请注意,所讨论的错误在多个方面都非常棘手。如果您(或 YouTube 应用程序)将嵌套的 Bundle 存储在彼此内部,则它们还需要设置适当的 ClassLoader ......有人应该告诉 Google 获得一些常识,只需在 Bundle 中使用 Thread#getContextClassLoader 一次解决该问题对于所有人。
视图状态保存和恢复的问题。 Youtube 应用程序有一个错误和存储视图状态,其中包含来自 youtube apk 的 class asc
实例。所以我们的应用程序无法恢复它,因为对此一无所知class。我的解决方案是通过下一个代码阻止 YoutubePlayerView
在 YouTubePlayerSupportFragment
中保存视图状态:
@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
的子 class 中。
此解决方案不会从捆绑包中删除 youtube 播放器状态。这样youtube播放器就恢复成功了。
我找到了适合我的解决方案。在我的 onCreate
方法中,我捕获了由于 youtube 播放器中的 ClassNotFoundException
而抛出的 RuntimeException
。视频失去了它的状态。按播放键会重新开始。否则照常营业。
我刚刚使用了一个适合我的 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);
}