Androidサービス内でGCM通知を受信する
-
21-12-2019 - |
質問
私は持っています Service
それは私のメインアプリケーションと一緒に実行されます。このサービスの目的は、バックグラウンドでメインサーバーと同期することです。更新のためにメインサーバーを常にポーリングしないようにするために、ダウンロードする変更があるときにアプリに通知するgoogle cloud messagingを実装しました。
私が置くとき GcmBroadcastReceiver
私のメインパッケージのクラスでは、通知を正常に受け取ります。しかし、私はこれを私のサービスによって受信したいので、それはそれからトリガーすることができます downloadChanges()
serivceの中のまた機能。私は本当にアプリが通知を拾うことを望んでいないし、使用する必要があります IPC
サービスと通信するには-それは少し複雑すぎるようです。
私のサービスがメインアプリの代わりにGCM通知を受け取る方法はありますか?
現在、私のコードは次のようなものです:
public class SyncService extends Service {
// lots of other stuff in this class
public class GcmBroadcastReceiver extends WakefulBroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Log.i(TAG, "GCM received!");
downloadChanges();
}
}
}
しかし、私は ClassNotFoundException
GCM通知を受信したとき。バックグラウンドサービスがGCM通知を処理する正しい方法は何ですか?
マニフェスト:
<application
android:name="com.appnl.myapp.CustomApp"
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:screenOrientation="portrait"
android:theme="@style/AppTheme" >
<meta-data android:name="com.google.android.gms.version"
android:value="@integer/google_play_services_version" />
<!-- android.name=".GcmBroadcastReceiver" -->
<receiver
android:name=".GcmBroadcastReceiver"
android:permission="com.google.android.c2dm.permission.SEND" >
<intent-filter>
<action android:name="com.google.android.c2dm.intent.RECEIVE" />
<category android:name="com.appnl.gcm" />
</intent-filter>
</receiver>
<service
android:name="service.SyncService"
android:enabled="true"
android:exported="false"
android:icon="@drawable/ic_launcher"
android:label="@string/service_name" >
<intent-filter>
<action android:name="service.SyncService" />
</intent-filter>
</service>
<!-- activity list -->
ログキャット:
04-11 12:40:19.474: E/AndroidRuntime(28044): FATAL EXCEPTION: main
04-11 12:40:19.474: E/AndroidRuntime(28044): java.lang.RuntimeException: Unable to instantiate receiver com.appnl.myapp.GcmBroadcastReceiver: java.lang.ClassNotFoundException: Didn't find class "com.appnl.myapp.GcmBroadcastReceiver" on path: /data/app/com.appnl.myapp-1.apk
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.app.ActivityThread.handleReceiver(ActivityThread.java:2493)
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.app.ActivityThread.access$1600(ActivityThread.java:159)
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1392)
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.os.Handler.dispatchMessage(Handler.java:99)
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.os.Looper.loop(Looper.java:137)
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.app.ActivityThread.main(ActivityThread.java:5419)
04-11 12:40:19.474: E/AndroidRuntime(28044): at java.lang.reflect.Method.invokeNative(Native Method)
04-11 12:40:19.474: E/AndroidRuntime(28044): at java.lang.reflect.Method.invoke(Method.java:525)
04-11 12:40:19.474: E/AndroidRuntime(28044): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1187)
04-11 12:40:19.474: E/AndroidRuntime(28044): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1003)
04-11 12:40:19.474: E/AndroidRuntime(28044): at dalvik.system.NativeStart.main(Native Method)
04-11 12:40:19.474: E/AndroidRuntime(28044): Caused by: java.lang.ClassNotFoundException: Didn't find class "com.appnl.myapp.GcmBroadcastReceiver" on path: /data/app/com.appnl.myapp-1.apk
04-11 12:40:19.474: E/AndroidRuntime(28044): at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:64)
04-11 12:40:19.474: E/AndroidRuntime(28044): at java.lang.ClassLoader.loadClass(ClassLoader.java:501)
04-11 12:40:19.474: E/AndroidRuntime(28044): at java.lang.ClassLoader.loadClass(ClassLoader.java:461)
04-11 12:40:19.474: E/AndroidRuntime(28044): at android.app.ActivityThread.handleReceiver(ActivityThread.java:2488)
04-11 12:40:19.474: E/AndroidRuntime(28044): ... 10 more
編集:私は私のサービスがaにあることを追加する必要があります セパレート 私のメインパッケージへのパッケージ。私はそれが問題の原因であると感じていますが、私は間違っている可能性があります。
解決
あなたの主な問題は、マニフェスト内の受信者の命名です。これは内部クラスであるため、外部クラス名を含め、その名前をappendedで区切られて追加する必要があります。例を挙げてみます。..
com.appnl.myapp.SyncService$GcmBroadcastReceiver
しかし、私はこれが物事を行うための良い方法であるかどうかはわかりません。
第一に、aのライフサイクル BroadcastReceiver
限りであります onReceive(...)
メソッドが実行され、そのメソッドは最小限の作業のみを行う必要があります。あなたの場合を除いて downloadChanges()
メソッドは、単にいくつかの並べ替えの非同期操作を開始します(ので、すぐに戻ります)あなたはブロックします onReceive(...)
メソッド。
第二に、あなたのアプローチは Service
永続的に実行されています。それが実際にそうであり、それが実行され続けることを保証できる場合、それは問題ありませんが、多くのバッテリーの消耗を危険にさらす可能性があ
あなたは永久に実行する必要がない場合 Service
その後、私は個人的に保つだろう BroadcastReceiver
分離して使用する IntentService
代わりに。あなたは、単に放送を受信し、その後でと思います onReceive(...)
使用 startService(...)
.これは保障します BroadcastReceiver
短い生命およびを持っています IntentService
独自のワーカースレッドを使用します(したがって、 NetworkOnMainThreadException
)また、作業が完了した後も自己終了します。