StrictMode для нижних версий платформы
-
25-09-2019 - |
Вопрос
Я начал использовать Android StrictMode и обнаружил, что было бы здорово, что он всегда работает во время развития, а не только на специальной ветви, которую я создал в Git. Причина, по которой я сделал это мои приложения для бега с 1,6 и вверх.
Я прочитал в блоге Android Developer, который вы можете настроить его, чтобы вы активировали его посредством отражения. Мне просто было интересно, как это на самом деле выглядело, и если бы можно было бы получить это документировать здесь (или где-то еще), а не иметь всех, кто хочет использовать, он работает.
Решение 2
Поэтому я не хотел ждать и решил приложить усилия и реализовать это сам. Он в основном сводится к упаковке StrictMode в классе обертки и выбирая во время выполнения через размышления, если мы сможем активировать его.
Я документировал его подробно в блоге пост и сделал это доступным в гадость.
Другие советы
Я прочитал сообщение в блоге Манфреда, но это не работает, если вы установите Версия целевой платформы ниже 2.3 потому что StrictMode.enableDefaults();
Метод недоступен.
Вот мое решение, которое полностью опирается на отражение и не генерирует ошибки компиляции:
try {
Class<?> strictModeClass = Class.forName("android.os.StrictMode", true, Thread.currentThread()
.getContextClassLoader());
Class<?> threadPolicyClass = Class.forName("android.os.StrictMode$ThreadPolicy", true, Thread
.currentThread().getContextClassLoader());
Class<?> threadPolicyBuilderClass = Class.forName("android.os.StrictMode$ThreadPolicy$Builder", true,
Thread.currentThread().getContextClassLoader());
Method setThreadPolicyMethod = strictModeClass.getMethod("setThreadPolicy", threadPolicyClass);
Method detectAllMethod = threadPolicyBuilderClass.getMethod("detectAll");
Method penaltyMethod = threadPolicyBuilderClass.getMethod("penaltyLog");
Method buildMethod = threadPolicyBuilderClass.getMethod("build");
Constructor<?> threadPolicyBuilderConstructor = threadPolicyBuilderClass.getConstructor();
Object threadPolicyBuilderObject = threadPolicyBuilderConstructor.newInstance();
Object obj = detectAllMethod.invoke(threadPolicyBuilderObject);
obj = penaltyMethod.invoke(obj);
Object threadPolicyObject = buildMethod.invoke(obj);
setThreadPolicyMethod.invoke(strictModeClass, threadPolicyObject);
} catch (Exception ex) {
Log.w(TAG, ex);
}
Я видел твой блог. Поскольку вы только когда-либо хотите настроить STRITHMODE максимум один раз на Java-файл, будет ли каким-либо смысл упростить код, чтобы вызвать настройку следующим образом?
Вот альтернативное strithmodewper:
import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.os.StrictMode;
public class StrictModeWrapper {
public static void init(Context context) {
// check if android:debuggable is set to true
int applicationFlags = context.getApplicationInfo().flags;
if ((applicationFlags & ApplicationInfo.FLAG_DEBUGGABLE) != 0) {
StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
.detectDiskReads()
.detectDiskWrites()
.detectNetwork()
.penaltyLog()
.build());
StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
.detectLeakedSqlLiteObjects()
.penaltyLog()
.penaltyDeath()
.build());
}
}
}
Из вашего кода вам нужно только сделать следующее:
try {
StrictModeWrapper.init(this);
}
catch(Throwable throwable) {
Log.v("StrictMode", "... is not available. Punting...");
}
куда это является локальным контекстом, таким как ваша деятельность или приложение или что-то еще. Это, кажется, работает для Android Pre-2.3, а также дает вам возможность использования других методов классов Builder для настройки StrictMode точно так, как вы хотите.
Я собрал еще одну вариацию на тему выше, которую я изложил в блог здесь. Отказ Основное отличие в моем подходе состоит в том, что он также предоставляет обертки для объектов диска и виртуальной политики, так что вы легко можете проводить кронштейн StrictMode-оскорбительный код с временными изменениями политики. Обратная связь Добро пожаловать.
К коду пикселя, я также добавил это (на основе StrictMode Android API API):
// VM policy
Class<?> VmPolicyClass = Class.forName("android.os.StrictMode$VmPolicy", true, Thread.currentThread().getContextClassLoader());
Class<?> VmPolicyBuilderClass = Class.forName("android.os.StrictMode$VmPolicy$Builder", true, Thread.currentThread().getContextClassLoader());
Method setVmPolicyMethod = strictModeClass.getMethod("setVmPolicy", VmPolicyClass);
Method detectLeakedSqlLiteObjectsMethod = VmPolicyBuilderClass.getMethod("detectLeakedSqlLiteObjects");
Method detectLeakedClosableObjectsMethod = null;
try
{
detectLeakedClosableObjectsMethod = VmPolicyBuilderClass.getMethod("detectLeakedClosableObjects");
}
catch (Exception e) {}
Method penaltyLogMethod = VmPolicyBuilderClass.getMethod("penaltyLog");
Method penaltyDeathMethod = VmPolicyBuilderClass.getMethod("penaltyDeath");
Method VmbuildMethod = VmPolicyBuilderClass.getMethod("build");
Constructor<?> VmPolicyBuilderConstructor = VmPolicyBuilderClass.getConstructor();
Object VmPolicyBuilderObject = VmPolicyBuilderConstructor.newInstance();
Object Vmobj = detectLeakedSqlLiteObjectsMethod.invoke(VmPolicyBuilderObject);
if (detectLeakedClosableObjectsMethod != null) Vmobj = detectLeakedClosableObjectsMethod.invoke(Vmobj);
Vmobj = penaltyLogMethod.invoke(Vmobj);
Vmobj = penaltyDeathMethod.invoke(Vmobj);
Object VmPolicyObject = VmbuildMethod.invoke(Vmobj);
setVmPolicyMethod.invoke(strictModeClass, VmPolicyObject);
Установите Android Mailest к чему-то вроде этого.
< uses-sdk android:minSdkVersion="8" android:targetSdkVersion="16" android:maxSdkVersion="16"/ >
Используйте код ниже в методе OnCreate.
int SDK_INT = android.os.Build.VERSION.SDK_INT;
} if (SDK_INT>8){
StrictMode.ThreadPolicy policy = new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
}
Примечание. Отключить предупреждение, поскольку вы уже проверяете, какая версия Android будет использовать этот код.
Этот код будет активирован, если версия Android выше, чем Android 2.2