使用GoogleCloudMessaging的Android gcm客户端

时间:2013-07-01 13:20:12

标签: android push-notification google-cloud-messaging

我正在使用使用GoogleCloudMessaging类使用gcm描述here的新方式。 但是10次中有9次给我SERVICE_NOT_AVAILABLE错误。只有一次注册设备。

07-01 14:24:55.907    2074-2090/com.deveyes.carrefour          W/System.err:                   java.io.IOException: SERVICE_NOT_AVAILABLE

07-01 14:24:56.037    2074-2090/com.deveyes.carrefour          W/System.err: at com.google.android.gms.gcm.GoogleCloudMessaging.register(Unknown Source)

我的活动中的代码如下:

 .....
     gcmRegId = gcm.register(CarrefourConfig.GCM_SENDER_ID);   
 .....                                                       

后来我才知道我需要使用自动重试。我看过this,但这份文件似乎已经过时了。与新方法一样,我们不再使用IntentService。还是我们?在BroadcastReciever里面,我可以收到消息并通知用户。

我在这里做错了什么?

编辑:

我的清单:

<?xml version="1.0" encoding="utf-8"?>

<uses-sdk
    android:minSdkVersion="8"
    android:targetSdkVersion="17" />


<supports-screens
        android:xlargeScreens="false"
 />


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

<uses-feature android:name="android.hardware.location" android:required="true" />
<uses-feature android:name="android.hardware.location.gps" android:required="false" />

<permission
        android:name="com.deveyes.carrefour.android.MAPS_RECEIVE"
        android:protectionLevel="signature" />


<uses-feature
        android:glEsVersion="0x00020000"
        android:required="true" />

<uses-permission android:name="com.deveyes.carrefour.android.permission.MAPS_RECEIVE" />
<uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />


<permission android:name="com.deveyes.carrefour.android.permission.C2D_MESSAGE"
            android:protectionLevel="signature" />
<uses-permission android:name="com.deveyes.carrefour.android.permission.C2D_MESSAGE" />

<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.WAKE_LOCK" />


<application
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/Theme.Carrefour"
        android:name=".CarrefourApplication">

    <meta-data
            android:name="com.google.android.maps.v2.API_KEY"
            android:value="AIzaSyCKznwauzk_rNdKf7Qw0rjpmf4AWchCRwI" />

    <activity
        android:name=".ui.MainActivity"
        android:label="@string/app_name"
        android:screenOrientation="portrait">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>


    <receiver
            android:name=".gcm.GCMBroadcastReceiver"
            android:permission="com.google.android.c2dm.permission.SEND" >
        <intent-filter>
            <action android:name="com.google.android.c2dm.intent.RECEIVE" />
            <category android:name="com.deveyes.carrefour" />
        </intent-filter>
    </receiver>

    <!--service android:name=".services.CarrefourLocationService" /-->


</application>

我的主要活动:

private void registerBackground() {

    new AsyncTask(){

        @Override
        protected Object doInBackground(Object[] objects) {

            try {
                if(gcm == null){
                    gcm = GoogleCloudMessaging.getInstance(context);
                }
                 gcmRegId = gcm.register(CarrefourConfig.GCM_SENDER_ID);

                Log.i(TAG,"regid:"+gcmRegId);

                setRegistrationId(context,gcmRegId);
                ServerUtilities.register(context,gcmRegId);


            } catch (IOException e) {
                e.printStackTrace();
            }
            return null;
        }
    }.execute(null,null,null);
}

3 个答案:

答案 0 :(得分:2)

问题在于:

您应该在以下所有3个地方使用相同的套餐名称,但在某些情况下您使用com.deveyes.carrefour.android,而在其他情况下则使用com.deveyes.carrefour

<permission android:name="com.deveyes.carrefour.android.permission.C2D_MESSAGE"
            android:protectionLevel="signature" />
<uses-permission android:name="com.deveyes.carrefour.android.permission.C2D_MESSAGE" />
...
    <receiver
            android:name=".gcm.GCMBroadcastReceiver"
            android:permission="com.google.android.c2dm.permission.SEND" >
        <intent-filter>
            <action android:name="com.google.android.c2dm.intent.RECEIVE" />
            <category android:name="com.deveyes.carrefour" />
        </intent-filter>
    </receiver>

如果您的包名称为com.deveyes.carrefour,请将清单更改为:

<permission android:name="com.deveyes.carrefour.permission.C2D_MESSAGE"
            android:protectionLevel="signature" />
<uses-permission android:name="com.deveyes.carrefour.permission.C2D_MESSAGE" />
...
    <receiver
            android:name=".gcm.GCMBroadcastReceiver"
            android:permission="com.google.android.c2dm.permission.SEND" >
        <intent-filter>
            <action android:name="com.google.android.c2dm.intent.RECEIVE" />
            <category android:name="com.deveyes.carrefour" />
        </intent-filter>
    </receiver>

答案 1 :(得分:1)

嘿,在GCM jar中应该有一个名为GCMRegistrar的东西你应该使用该方法注册到GCM。以下是代码,试试吧

GCMRegistrar.checkDevice(this);
 GCMRegistrar.checkManifest(this);
registerReceiver(mHandleMessageReceiver, new IntentFilter(
                DISPLAY_MESSAGE_ACTION));

        final String regId = GCMRegistrar.getRegistrationId(this);
        if (regId.equals("")) {
             GCMRegistrar.register(this, SENDER_ID); //this is where it registers
        } else {

            if (GCMRegistrar.isRegisteredOnServer(this)) {
                   Toast.makeText(getApplicationContext(), "Already registered with GCM", Toast.LENGTH_LONG).show();
            } 

我希望它适合你

答案 2 :(得分:0)

我遇到了同样的问题。

9/20抛出SERVICE_NOT_AVAILABLE异常。显然这是谷歌方面的一个错误。适用于某些设备(S3)而不适用于其他设备(LG L5)。

检查Here

我认为要做的是暂时使用旧的API,直到问题得到解决。