我在更新 Android 应用时,调用 URL.setURLStreamHandlerFactory(factory);
导致了一个意料之外的错误。
public class ApplicationRoot extends Application {
static {
/* Add application support for custom URI protocols. */
final URLStreamHandlerFactory factory = new URLStreamHandlerFactory() {
@Override
public URLStreamHandler createURLStreamHandler(final String protocol) {
if (ExternalProtocol.PROTOCOL.equals(protocol)) {
return new ExternalProtocol();
}
if (ArchiveProtocol.PROTOCOL.equals(protocol)) {
return new ArchiveProtocol();
}
return null;
}
};
URL.setURLStreamHandlerFactory(factory);
}
}
简介:
我的情况如下:我正在维护一个用于企业应用的非市场应用程序。 我的公司销售预装有由公司开发和维护的应用程序的平板电脑。 这些预装的应用程序不是ROM的一部分,它们像典型的“未知来源”应用程序一样安装。 我们不通过Play商店或任何其他市场执行更新。 相反,应用程序更新由自定义的“更新管理器”应用程序控制,该应用程序直接与我们的服务器通信以执行OTA更新。
问题:
我正在维护的这个更新管理器应用程序偶尔需要更新自身。 在应用程序自我更新后,它将通过android.intent.action.PACKAGE_REPLACED
广播重新启动,我在AndroidManifest中注册了此广播。 然而,在更新后立即重新启动应用程序时,我偶尔会收到此Error
信息。
java.lang.Error: Factory already set
at java.net.URL.setURLStreamHandlerFactory(URL.java:112)
at com.xxx.xxx.ApplicationRoot.<clinit>(ApplicationRoot.java:37)
at java.lang.Class.newInstanceImpl(Native Method)
at java.lang.Class.newInstance(Class.java:1208)
at android.app.Instrumentation.newApplication(Instrumentation.java:996)
at android.app.Instrumentation.newApplication(Instrumentation.java:981)
at android.app.LoadedApk.makeApplication(LoadedApk.java:511)
at android.app.ActivityThread.handleReceiver(ActivityThread.java:2625)
at android.app.ActivityThread.access$1800(ActivityThread.java:172)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1384)
at android.os.Handler.dispatchMessage(Handler.java:102)
at android.os.Looper.loop(Looper.java:146)
at android.app.ActivityThread.main(ActivityThread.java:5653)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1291)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1107)
at dalvik.system.NativeStart.main(Native Method)
请注意,大多数情况下,应用程序会正常重新启动。 但是,偶尔会出现上述错误。我感到困惑,因为我唯一调用setURLStreamHandlerFactory
的地方就在这里,并且是在static
块中完成的,我认为(如果我错了,请纠正我),只有在首次加载ApplicationRoot
类时才会调用它一次。但是,似乎它被调用了两次,导致出现上述错误。
问题:
究竟是怎么回事?我目前唯一的猜测是,对于正在更新的应用程序,更新后的VM /进程与正在更新的先前安装的应用程序相同,因此当调用新ApplicationRoot
的static
块时,由旧ApplicationRoot
设置的URLStreamHandlerFactory
仍然处于“活动状态”。这可能吗?我该如何避免这种情况?看到它并不总是发生,似乎是某种竞争条件;也许是Android的APK安装例程内部的问题?谢谢,
编辑:
按要求附加代码。这是处理广播的清单部分。<receiver android:name=".OnSelfUpdate" >
<intent-filter>
<action android:name="android.intent.action.PACKAGE_REPLACED" />
<data android:scheme="package" />
</intent-filter>
</receiver>
并且BroadcastReceiver
本身
public class OnSelfUpdate extends BroadcastReceiver {
@Override
public void onReceive(final Context context, final Intent intent) {
/* Get the application(s) updated. */
final int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
final PackageManager packageManager = context.getPackageManager();
final String[] packages = packageManager.getPackagesForUid(uid);
if (packages != null) {
final String thisPackage = context.getPackageName();
for (final String pkg : packages) {
/* Check to see if this application was updated. */
if (pkg.equals(thisPackage)) {
final Intent intent = new Intent(context, MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
break;
}
}
}
}
}