Фоновые геозоны действительно работают на Android 8+? - PullRequest
0 голосов
/ 01 марта 2019

У меня есть приложение геозон, которое я пытаюсь перенести, чтобы начать работать на Android 8+.Я прочитал учебник по теме и переключился на использование compile 'com.google.android.gms:play-services-location:16.0.0'.

События входа в геозону никогда не запускаются, когда приложение не на переднем плане. Я жду больше, чем две минуты, которые, по словам доктора, могут занять.Я ждал до 15 минут безрезультатно.Как только я вывожу приложение на передний план внутри геозоны, оно сразу же срабатывает.

Я понимаю, что на Android 8+ существуют ограничения на фоновые сервисы, но в руководстве Google говорится об использовании IntentService, который долженбыть заблокированным на Android 8+. РЕДАКТИРОВКА ОБРАТНОЙ СВЯЗИ: Вышеупомянутое руководство абсолютно неверно.Не следуйте за этим.Служба IntentService не будет работать.

Я попытался выполнить этот ответ , в котором говорится, что новый GeofencingClient используется для настройки ваших намерений решить эту проблему.(Я не понимаю, почему это помогло бы, потому что он все еще использует IntentService для получения событий, и он не работает в моем коде.)

Этот ответ на связанный вопрос здесь предполагает, что вы должны использовать BroadcastReceiver вместо IntentService, но я не понимаю, почему это также будет иметь значение, поскольку Android 8 накладывает те же ограничения на BroadcastReceivers в фоновом режиме, что и IntentServices. ОБРАТНАЯ СВЯЗЬ: Этот связанный ответ также является правильным.Хотя неявные BroadcastReceivers ограничены, явные, которые регистрируются во время выполнения для отправки запросов в определенный пакет, не ограничены и работают должным образом.

Действительно ли возможно получить обратные вызовы геозон, которые пробуждают ваше приложение вфон на Android 8+?Если да, то как вы можете заставить это работать?

Моя настройка геозон:

googleApiClient = new GoogleApiClient.Builder(this.context)
        .addApi(LocationServices.API)
        .addConnectionCallbacks(getServiceSetup())
        .addOnConnectionFailedListener(getServiceFailureStrategy())
        .build();
geofencingClient = LocationServices.getGeofencingClient(context);

Intent intent =  new Intent(context, mIntentService);
// We use FLAG_UPDATE_CURRENT so that we get the same pending intent back when
// calling addGeofences() and removeGeofences().
geofencePendingIntent = PendingIntent.getService(context, 0, intent, PendingIntent.
            FLAG_UPDATE_CURRENT);

geofencingClient.addGeofences(geofences, geofencePendingIntent)
                .addOnSuccessListener(new OnSuccessListener<Void>() {
                 @Override
                   public void onSuccess(Void aVoid) {
                     getAddGeofencesCallback(runnable, geofencesToAdd).onResult(new Status(0));
                     Log.e(TAG, "Successfully added geofences with GeofencingClient");
                   }
                 })
                .addOnFailureListener(new OnFailureListener() {
                   @Override
                   public void onFailure(Exception e) {
                     getAddGeofencesCallback(runnable, geofencesToAdd).onResult(new Status(1));
                     Log.e(TAG, "Failed to add geofences with GeofencingClient", e);

IntentService:

public class GeofenceService extends IntentService {
    private static final String TAG = "GeofenceService";

    public GeofenceService() {
        super(TAG);
    }

    @Override
    protected void onHandleIntent(Intent intent) {
        GeofenceTransition transition = GeofenceServiceManager.getGeofenceTransition(intent);
        if (transition == null) {
            return;
        }

        GeofenceServiceManager.logd(
                TAG,
                "onHandleIntent. geofence: " +
                        GeofenceMessage.notificationTitleFor(this, transition)
        );
        processErrors(transition);
        sendBroadcast(transition);
    }

    private Intent generateBroadcast(GeofenceTransition transition) {
        Intent broadcastIntent = new Intent();

        broadcastIntent.addCategory(GeofenceUtils.CATEGORY_GEOFENCE_SERVICES)
                       .setAction(GeofenceUtils.actionNameFor(transition))
                       .putStringArrayListExtra(
                               GeofenceUtils.GEOFENCE_IDS,
                               new ArrayList<String>(transition.getIds())
                                               );

        return broadcastIntent;
    }

    private void sendBroadcast(GeofenceTransition transition) {
        if (transition.isError() || transition.unknownType()) {
            GeofenceServiceManager.logd(TAG, "geofence transition is error or unknown type.");
            return;
        }

        broadcastManager().sendBroadcast(generateBroadcast(transition));
    }

    private LocalBroadcastManager broadcastManager() {
        return LocalBroadcastManager.getInstance(this);
    }

    private void processErrors(GeofenceTransition transition) {
        if (!transition.isError()) {
            return;
        }

        Log.e(TAG, "geofence error: "+GeofenceMessage.errorDetailsFor(this, transition));

        Intent broadcastIntent = generateBroadcast(transition);
        broadcastIntent.putExtra(
                GeofenceUtils.GEOFENCE_STATUS,
                GeofenceMessage.errorMessageFor(this, transition)
                                );
        broadcastManager().sendBroadcast(broadcastIntent);
    }
}

AndroidManifest.xml:

...
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />

      <service
          android:name=".geofence.GeofenceService"
          android:enabled="true"
          android:exported="false">
      </service>
...

РЕДАКТИРОВАТЬ 1: На основании задокументированных Google ограничений на фоновую обработку Android 8+, мне кажется, что это невозможно.Android 8+ убивает запущенные приложения в течение 10 минут после входа в фоновый режим и блокирует запуск сервисов, включая сервисы намерений, когда не на переднем плане.Тем не менее, Google говорит:

Примечание. На Android 8.0 (уровень API 26) и выше, если приложение работает в фоновом режиме и отслеживает геозону, то устройство реагирует на события геозонирования каждые пару минут.,Чтобы узнать, как адаптировать ваше приложение к этим пределам отклика, см. Фоновые ограничения местоположения.

Как это возможно с задокументированными фоновыми ограничениями выполнения?

РЕДАКТИРОВАТЬ 2: Я видел BroadcastReceiver, объявленный в манифесте для запуска приложения в фоновом режиме на Android 8+, используя намеренное обнаружение Bluetooth LE.Этот код выглядит следующим образом:

Intent intent = new Intent();
intent.setComponent(new ComponentName(context.getPackageName(), "com.example.MyBroadcastReceiver"));
<receiver android:name="com.example.MyBroadcastReceiver">
         </receiver>

Это работает для обнаружения BLE для запуска приложений в фоновом режиме.Являются ли системные цели, подобные приведенным выше, каким-то образом защищены от пределов фонового выполнения Android 8+ (в отличие от IntentServices) при доставке в BroadcastReceiver?Если да, будет ли это работать с геозонами? (Оповещение спойлера: Да, будет! Прочитайте принятый ответ ниже.)

Ответы [ 2 ]

0 голосов
/ 06 марта 2019

Вам необходимо заменить:

geofencePendingIntent = PendingIntent.getService(context, 0, intent, PendingIntent.
            FLAG_UPDATE_CURRENT); 

на

Intent intent = new Intent(this, GeofenceBroadcastReceiver.class); 
mGeofencePendingIntent = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);

Для поддержки Android 8 вам следует использовать BroadcastReceiver для получения намерения геозоны, поскольку Фоновые пределы выполнения (служба не может быть запущена в фоновом режиме)

см. Более подробную информацию: https://github.com/googlesamples/android-play-location/commit/5f83047c8a462d7c619f6275b624e219b4622322

Пример использования Google Geofence: https://github.com/googlesamples/android-play-location/tree/master/Geofencing

0 голосов
/ 04 марта 2019

Если вы доверяете себе, что все делаете правильно, в этой статье вы можете использовать настройку ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS .Этот параметр отключает оптимизацию батареи приложения.Таким образом, намерения не блокируются системой.Потому что иногда система блокирует нашу работу для увеличения срока службы батареи.Так что если ваше приложение работает с этим параметром, вы можете обвинить систему.

Еще одна вещь.Как вы тестировали геозону?

...