在Android服务中接收GCM通知

时间:2014-04-11 11:24:33

标签: android android-service google-cloud-messaging

我有一个Service与我的主应用程序一起运行。此服务的目的是在后台与主服务器同步。为了避免不断轮询主服务器进行更新,我已经实施了Google云消息传递,以便在有下载更改时通知应用。

当我将GcmBroadcastReceiver类放在我的主程序包中时,它会收到通知。但是,我希望我的服务能够接收到这一点,因此它可以在服务中触发downloadChanges()功能。我真的不希望应用程序接收通知,然后必须使用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();
        }
    }
}

但是,我在收到GCM通知时收到ClassNotFoundException。我的后台服务处理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 -->

logcat的:

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

编辑:我应该将我的服务添加到我的主程序包的单独的包中。我有一种感觉是导致问题的原因,但我可能是错的。

1 个答案:

答案 0 :(得分:1)

您的主要问题是清单中接收者的命名。因为它是一个内部类,所以需要包含外部类名,并将其名称附加为$。实施例...

com.appnl.myapp.SyncService$GcmBroadcastReceiver

但是,我不确定这是做事的好方法。

首先BroadcastReceiver的生命周期与onReceive(...)方法执行一样长,该方法应该只做最少的工作。除非您的downloadChanges()方法只是启动某种异步操作(因此立即返回),否则您将阻止onReceive(...)方法。

其次,您的方法假定Service永久运行。如果情况确实如此,你可以保证它会继续运行那么那很好,但你可能会冒很多电池消耗的风险。

如果您不需要永久运行Service,那么我个人会将BroadcastReceiver分开并改为使用IntentService。然后,您只需接收广播,然后在onReceive(...)中使用startService(...)。这将确保BroadcastReceiver的生命周期短,IntentService使用自己的工作线程(从而避免使用NetworkOnMainThreadException),并且在完成工作后也会自行终止。< / p>