setURLStreamHandlerFactory and “java.lang.Error: Factory already set”
我遇到了在正在更新的Android应用程序中调用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | 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商店或任何其他市场进行更新。相反,应用程序更新由自定义Update Manager应用程序控制,该应用程序直接与我们的服务器通信以执行OTA更新。
问题:
我维护的此Update Manager应用程序偶尔需要自行更新。在应用程序更新后,它立即通过
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | 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) |
请注意,大多数情况下,应用程序正常重新启动。但是,每隔一段时间,我就会得到上述错误。我很困惑,因为我唯一称之为
题:
炽热的山姆在发生什么?我在这一点上唯一的猜测是更新后的应用程序的VM /进程与之前安装的应用程序相同,因此当调用新
编辑:
要求的附加代码。这是处理广播的清单部分
1 2 3 4 5 6 | <receiver android:name=".OnSelfUpdate"> <intent-filter> <data android:scheme="package" /> </intent-filter> </receiver> |
而
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | 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; } } } } } |
加载类时执行静态块 - 如果由于某种原因重新加载类(例如,当它被更新时),它将再次执行。
在您的情况下,这意味着您设置上次加载的
除非您更新
有两种方法可以解决这个问题:
抓住
实现一个非常简单的包装器,该包装器委托给您可以替换的另一个
跟踪您是否已使用系统属性安装了处理程序。
码:
1 2 3 4 5 6 | public static void maybeInstall(URLStreamHandlerFactory factory) { if(System.getProperty("com.xxx.streamHandlerFactoryInstalled") == null) { URL.setURLStreamHandlerFactory(factory); System.setProperty("com.xxx.streamHandlerFactoryInstalled","true"); } } |
码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | public static void forcefullyInstall(URLStreamHandlerFactory factory) { try { // Try doing it the normal way URL.setURLStreamHandlerFactory(factory); } catch (final Error e) { // Force it via reflection try { final Field factoryField = URL.class.getDeclaredField("factory"); factoryField.setAccessible(true); factoryField.set(null, factory); } catch (NoSuchFieldException | IllegalAccessException e1) { throw new Error("Could not access factory field on URL class: {}", e); } } } |
Oracle的
AFAIK您可以/不应该重启JVM。此外,正如您已经发现的那样,您无法在JVM中为单个应用程序设置两次
您的应用程序应该尝试仅在不是时设置工厂:
如果您的应用程序更新还包括更新工厂,您可以尝试终止您的应用程序所在的进程,但我不这样做是一个好主意,更糟糕的是 - 它可能甚至不起作用。