SetURLStreamHandlerFactory и "java.lang.Error: Factory уже установлен"

Я столкнулся с непредвиденной ошибкой, вызванной вызовом 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);
    }

}

Введение:

Вот моя ситуация: я поддерживаю нерыночное приложение, используемое на предприятии. Мой бизнес продает планшеты с предварительно установленными приложениями, которые разрабатываются и поддерживаются бизнесом. Эти предустановленные приложения не являются частью ПЗУ; они устанавливаются как типичные приложения Неизвестного источника. Мы не производим обновления через Play Store или на любом другом рынке. Скорее, обновления приложений контролируются специальным приложением диспетчера обновлений, которое напрямую связывается с нашими серверами для выполнения обновлений 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/процесс для обновленного приложения совпадает с ранее установленным приложением, которое обновляется, поэтому, когда вызывается блок static для нового ApplicationRoot, набор URLStreamHandlerFactory по старой ApplicationRoot по-прежнему "активен". Это возможно? Как я могу избежать этой ситуации? Увидев, что это не всегда происходит, это похоже на какое-то гоночное состояние; возможно, в программе Android APK? Спасибо,

Edit:

Дополнительный код в соответствии с запросом. Вот часть манифеста, связанная с вещанием

<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;
                }
            }
        }
    }

}

Ответы

Ответ 1

Статические блоки выполняются при загрузке класса - если класс по какой-либо причине перезагружается (например, когда он был обновлен), он будет выполнен снова.

В вашем случае это означает, что URLStreamHandlerFactory, установленный в предыдущее время, когда он был загружен, останется.

Это не проблема, если вы не обновили URLStreamHandlerFactory.

Есть два способа исправить это:

  • Поймайте Error и продолжайте свой веселый путь, игнорируя тот факт, что вы все еще используете старый factory.

  • Внедрите очень простую оболочку, которая делегирует другому 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");
    }
}
  1. Принудительная замена с использованием отражения. Я понятия не имею, почему вы можете установить только 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);
        }
    }
}

Имя поля factory в Oracle JRE, может отличаться на Android.

Ответ 2

AFAIK вы можете/не должны перезапускать JVM. Кроме того, как вы уже выяснили, вы не можете установить URLStreamHandlerFactory дважды в JVM для одного приложения.

Ваше приложение должно попытаться установить factory только в том случае, если это не так:

try {
    URL.setURLStreamHandlerFactory(factory);
} catch (Error e) {
    e.printStackTrace();
}

Если обновления вашего приложения также включают обновление factory, вы можете попробовать убить процесс, в котором находится ваше приложение, но я не очень рекомендую это делать так что еще хуже - он может даже не работать.