我遇到了一个因URL.setURLStreamHandlerFactory(factory);正在更新的Android应用程序中调用而导致的意外错误.
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);
}
}
Run Code Online (Sandbox Code Playgroud)
介绍:
这是我的情况:我正在维护一种用于企业时尚的非市场应用程序.我的公司销售的平板电脑包含由业务开发和维护的预安装应用程序.这些预安装的应用程序不是ROM的一部分; 它们作为典型的未知源应用程序安装.我们不会通过Play商店或任何其他市场进行更新.相反,应用程序更新由自定义Update Manager应用程序控制,该应用程序直接与我们的服务器通信以执行OTA更新.
问题:
我维护的此Update Manager应用程序偶尔需要自行更新.应用程序更新后,它立即通过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)
Run Code Online (Sandbox Code Playgroud)
请注意,大多数情况下,应用程序正常重新启动.但是,每隔一段时间,我就会得到上述错误.我很困惑,因为我唯一的地方setURLStreamHandlerFactory就是在这里,它是在一个static区块中完成的,我认为 - 虽然如果我错了就纠正我 - 只有一次调用,ApplicationRoot如果第一次加载的话.但是,它似乎被调用两次,导致上述错误.
题:
在什么炽热的SAMS是怎么回事?我唯一的猜测是更新的应用程序的VM /进程与之前安装的应用程序相同,因此当调用new的static块时,旧的设置仍然是"活动的".这可能吗?我该如何避免这种情况?看到它并不总是发生,它似乎是某种竞争条件; 也许在Android的APK安装程序中?谢谢, ApplicationRootURLStreamHandlerFactory ApplicationRoot
编辑:
要求的附加代码.这是处理广播的清单部分
<receiver android:name=".OnSelfUpdate" >
<intent-filter>
<action android:name="android.intent.action.PACKAGE_REPLACED" />
<data android:scheme="package" />
</intent-filter>
</receiver>
Run Code Online (Sandbox Code Playgroud)
而它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;
}
}
}
}
}
Run Code Online (Sandbox Code Playgroud)
加载类时执行静态块 - 如果由于某种原因重新加载类(例如,当它被更新时),它将再次执行.
在您的情况下,这意味着URLStreamHandlerFactory您设置上次加载的时间将保留.
除非你更新了,否则这不是一个真正的问题URLStreamHandlerFactory.
有两种方法可以解决这个问题:
抓住Error并继续你的快乐方式,忽略你仍在使用旧工厂的事实.
实现一个非常简单的包装器,该包装器委托给URLStreamHandlerFactory您可以替换的另一个包装器,并且您不必更改.你将在这里遇到与包装器相同的问题,所以你需要抓住Error那个或者将它与选项3结合起来.
跟踪您是否已使用系统属性安装了处理程序.
码:
public static void maybeInstall(URLStreamHandlerFactory factory) {
if(System.getProperty("com.xxx.streamHandlerFactoryInstalled") == null) {
URL.setURLStreamHandlerFactory(factory);
System.setProperty("com.xxx.streamHandlerFactoryInstalled", "true");
}
}
Run Code Online (Sandbox Code Playgroud)
URLStreamHandlerFactory一次 - 这对我来说没什么意义TBH.码:
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);
}
}
}
Run Code Online (Sandbox Code Playgroud)
字段名称factory在Oracle上JRE,在Android上可能有所不同.
AFAIK 你可以/不应该重新启动 JVM。此外,正如您已经发现的,您不能URLStreamHandlerFactory在 JVM 中为单个应用程序设置两次。
您的应用程序应该仅在以下情况下尝试设置工厂:
try {
URL.setURLStreamHandlerFactory(factory);
} catch (Error e) {
e.printStackTrace();
}
Run Code Online (Sandbox Code Playgroud)
如果您的应用程序更新还包括更新工厂,您可以尝试终止您的应用程序所在的进程,但我不认为这样做是一个好主意,更糟糕的是 - 它甚至可能不起作用。
| 归档时间: |
|
| 查看次数: |
4238 次 |
| 最近记录: |