web-dev-qa-db-ja.com

サービスインテントの開始は許可されていません-Android Oreo

現在、OreoでクラッシュするstartWakefulService関数を使用しています。 startForegroundService()に切り替えてフォアグラウンドサービスを使用するか、JobIntentServiceに切り替える必要があることに気付きましたが、以下のコードに基づいて何をすべきかわかりません。 (申し訳ありませんが、Android初心者です。)正しい方向に向けてのポイントは大歓迎です。

public class GcmBroadcastReceiver extends WakefulBroadcastReceiver {

@Override
public void onReceive(Context context, Intent intent) {

    // Explicitly specify that GCMIntentService will handle the intent.
    ComponentName comp = new ComponentName(context.getPackageName(), GCMIntentService.class.getName());
    // Start the service, keeping the device awake while it is launching.
    startWakefulService(context, (intent.setComponent(comp)));

    setResultCode(Activity.RESULT_OK);
}
}

これは、Android 8.xで実行したときに発生する現在のエラーです。

致命的な例外:Java.lang.RuntimeExceptionレシーバーを起動できませんcom.heyjude.heyjudeapp.gcm.GcmBroadcastReceiver:Java.lang.IllegalStateException:サービスの開始を許可していません{act = com.google.Android.c2dm.intent.RECEIVE flg = 0x1000010 pkg = com.app.app cmp = com.app.app/.gcm.GCMIntentService(extras)}:アプリはバックグラウンドuid UidRecordにあります{f602fba u0a114 RCVR idle procs:1 seq(0,0,0)}

4
Marcus Smith

私は同じ行動を経験します。

なぜこの問題が発生するのですか?

新しい バックグラウンド実行制限 Android 8)のため、サービスバックグラウンドで開始しないでください。

修正方法

GCMIntetServiceの代わりにJobIntentServiceIntentServiceに移行します。

次の手順に従ってください:1)BIND_JOB_SERVICE権限をサービスに追加します。

<service Android:name=".service.GCMIntentService"
        Android:exported="false"
        Android:permission="Android.permission.BIND_JOB_SERVICE"/>

2)GCMIntentServiceでは、IntentServiceを拡張する代わりに、Android.support.v4.app.JobIntentServiceを使用してonHandleWorkをオーバーライドし、overrideonHandleIntentから削除します。

public class GCMIntentService extends JobIntentService {

    // Service unique ID
    static final int SERVICE_JOB_ID = 50;

    // Enqueuing work in to this service.
    public static void enqueueWork(Context context, Intent work) {
        enqueueWork(context, GCMIntentService.class, SERVICE_JOB_ID, work);
    }

    @Override
    protected void onHandleWork(@NonNull Intent intent) {
        onHandleIntent(intent);
    }

    private void onHandleIntent(Intent intent) {
        //Handling of notification goes here
    }
}

最後に、GCMBroadcastReceiverGCMIntentServiceをエンキューします。

public class GCMBroadcastReceiver extends WakefulBroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        // Explicitly specify that GcmIntentService will handle the intent.
        ComponentName comp = new ComponentName(context.getPackageName(),
                GCMIntentService.class.getName());
        // Start the service, keeping the device awake while it is launching.
        // startWakefulService(context, (intent.setComponent(comp)));

        //setResultCode(Activity.RESULT_OK);

        GCMIntentService.enqueueWork(context, (intent.setComponent(comp)));
    }
}

この実装は、ターゲットSDKを27に更新した後で機能します。

16

documentation に基づく:

Android O以降、バックグラウンドチェックの制限により、このクラスは一般的に役立ちません。(保証がないため、ブロードキャストの受信からサービスを開始することは一般的に安全ではありません。アプリがこの時点でフォアグラウンドにあるため、そうすることが許可されています。)代わりに、開発者はAndroid.app.job.JobSchedulerを使用してジョブをスケジュールする必要があります。その間、アプリはウェイクロックを保持します(ジョブのウェイクロックの保持はシステムが処理します)。

唯一の代替策は、ForeGroundServiceを開始するか、JobScheduler/JobIntentServiceを使用することです

次のようにフォアグラウンドサービスを開始できます:

_@Override
public void onReceive(Context context, Intent intent) {

    // Explicitly specify that GCMIntentService will handle the intent.
    ComponentName comp = new ComponentName(context.getPackageName(), GCMIntentService.class.getName());
    // Start the service, keeping the device awake while it is launching.
    ContextCompat.startForegroundService(context,intent.setComponent(comp));
    ...
}
_

また、ServiceクラスからstartService()を呼び出して、通知を表示する必要もあります。実装の詳細については、これに対する私の答え [〜#〜] so [〜#〜] をフォローできます。

1
Sagar