1

初めて Samsung Galaxy ace2 でアプリを実行できました。現在、GCM 登録に失敗して次のエラーが発生しています。エミュレータでは正常に動作しますが、そのデバイスでは動作しません。以下は、エラーとGCMIntentService.java.

エラー:

03-21 09:25:33.110: V/GCMBaseIntentService(6018): Acquiring wakelock
03-21 09:25:33.120: V/GCMBaseIntentService(6018): Intent service name: GCMIntentService-1089764589011-11
03-21 09:25:33.130: D/GCMBaseIntentService(6018): handleRegistration: registrationId = null, error = SERVICE_NOT_AVAILABLE, unregistered = null
03-21 09:25:33.130: D/GCMBaseIntentService(6018): Registration error: SERVICE_NOT_AVAILABLE
03-21 09:25:33.130: I/GCMIntentService(6018): Received recoverable error: SERVICE_NOT_AVAILABLE
03-21 09:25:33.130: D/GCMBaseIntentService(6018): Scheduling registration retry, backoff = 98657 (96000)
03-21 09:25:33.200: V/GCMBaseIntentService(6018): Releasing wakelock
03-21 09:26:42.950: D/dalvikvm(6018): GC_CONCURRENT freed 354K, 48% free 3310K/6279K, external 630K/1286K, paused 7ms+9ms
03-21 09:27:11.800: V/GCMBroadcastReceiver(6018): onReceive: com.google.android.gcm.intent.RETRY
03-21 09:27:11.800: V/GCMBroadcastReceiver(6018): GCM IntentService class: com.dorji.finalproject.GCMIntentService
03-21 09:27:11.800: V/GCMBaseIntentService(6018): Acquiring wakelock
03-21 09:27:11.830: V/GCMBaseIntentService(6018): Intent service name: GCMIntentService-1089764589011-12
03-21 09:27:11.840: V/GCMRegistrar(6018): Registering app com.dorji.finalproject of senders 1089764589011
03-21 09:27:11.840: V/GCMBaseIntentService(6018): Releasing wakelock
03-21 09:27:12.010: V/GCMBroadcastReceiver(6018): onReceive: com.google.android.c2dm.intent.REGISTRATION
03-21 09:27:12.010: V/GCMBroadcastReceiver(6018): GCM IntentService class: com.dorji.finalproject.GCMIntentService
03-21 09:27:12.010: V/GCMBaseIntentService(6018): Acquiring wakelock
03-21 09:27:12.020: V/GCMBaseIntentService(6018): Intent service name: GCMIntentService-1089764589011-13
03-21 09:27:12.020: D/GCMBaseIntentService(6018): handleRegistration: registrationId = null, error = SERVICE_NOT_AVAILABLE, unregistered = null
03-21 09:27:12.020: D/GCMBaseIntentService(6018): Registration error: SERVICE_NOT_AVAILABLE
03-21 09:27:12.020: I/GCMIntentService(6018): Received recoverable error: SERVICE_NOT_AVAILABLE
03-21 09:27:12.020: D/GCMBaseIntentService(6018): Scheduling registration retry, backoff = 105051 (192000)
03-21 09:27:12.070: V/GCMBaseIntentService(6018): Releasing wakelock

GCMIntentService.java

package com.dorji.finalproject;

import static com.dorji.finalproject.CommonUtilities.SENDER_ID;
import static com.dorji.finalproject.CommonUtilities.displayMessage;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

import com.dorji.finalproject.R;
import com.google.android.gcm.GCMBaseIntentService;

public class GCMIntentService extends GCMBaseIntentService {

    private static final String TAG = "GCMIntentService";

    public GCMIntentService() {
        super(SENDER_ID);
    }

    /**
     * Method called on device registered
     **/
    @Override
    protected void onRegistered(Context context, String registrationId) {
        Log.i(TAG, "Device registered: regId = " + registrationId);
        displayMessage(context, "Your device registred with GCM");
        ServerUtilities.register(context, registrationId);
    }

    /**
     * Method called on device un registred
     * */
    @Override
    protected void onUnregistered(Context context, String registrationId) {
        Log.i(TAG, "Device unregistered");
        displayMessage(context, getString(R.string.gcm_unregistered));
        ServerUtilities.unregister(context, registrationId);
    }

    /**
     * Method called on Receiving a new message
     * */
    @Override
    protected void onMessage(Context context, Intent intent) {
        Log.i(TAG, "Received message");
        String message = intent.getExtras().getString("message");

        displayMessage(context, message);
        // notifies user
        generateNotification(context, message);

    }

    /**
     * Method called on receiving a deleted message
     * */
    @Override
    protected void onDeletedMessages(Context context, int total) {
        Log.i(TAG, "Received deleted messages notification");
        String message = getString(R.string.gcm_deleted, total);
        displayMessage(context, message);
        // notifies user
        generateNotification(context, message);
    }

    /**
     * Method called on Error
     * */
    @Override
    public void onError(Context context, String errorId) {
        Log.i(TAG, "Received error: " + errorId);
        displayMessage(context, getString(R.string.gcm_error, errorId));
    }

    @Override
    protected boolean onRecoverableError(Context context, String errorId) {
        // log message
        Log.i(TAG, "Received recoverable error: " + errorId);
        displayMessage(context, getString(R.string.gcm_recoverable_error,
                errorId));
        return super.onRecoverableError(context, errorId);
    }

    /**
     * Issues a notification to inform the user that server has sent a message.
     */
    private static void generateNotification(Context context, String message) {
        int icon = R.drawable.ic_launcher;
        long when = System.currentTimeMillis();
        NotificationManager notificationManager = (NotificationManager)
                context.getSystemService(Context.NOTIFICATION_SERVICE);
        Notification notification = new Notification(icon, message, when);

        String title = context.getString(R.string.app_name);

        Intent notificationIntent = new Intent(context, MainActivity.class);
        // set intent so it does not start a new activity
        notificationIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP |
                Intent.FLAG_ACTIVITY_SINGLE_TOP);
        PendingIntent intent =
                PendingIntent.getActivity(context, 0, notificationIntent, 0);
        notification.setLatestEventInfo(context, title, message, intent);
        notification.flags |= Notification.FLAG_AUTO_CANCEL;

        // Play default notification sound
        notification.defaults |= Notification.DEFAULT_SOUND;

        //notification.sound = Uri.parse("android.resource://" + context.getPackageName() + "your_sound_file_name.mp3");

        // Vibrate if vibrate is enabled
        notification.defaults |= Notification.DEFAULT_VIBRATE;
        notificationManager.notify(0, notification);      

    }

}

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.dorji.finalproject"
    android:versionCode="1"
    android:versionName="1.0" >

    <!-- GCM requires Android SDK version 2.2 (API level 8) or above. -->
    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="16" />

    <supports-screens android:largeScreens="true"
        android:normalScreens="true" android:smallScreens="true"
        android:resizeable="true" android:anyDensity="true" />

    <!-- GCM connects to Internet Services. -->
    <uses-permission android:name="android.permission.INTERNET" />

    <!-- GCM requires a Google account. -->
    <uses-permission android:name="android.permission.GET_ACCOUNTS" />

    <!-- Keeps the processor from sleeping when a message is received. -->
    <uses-permission android:name="android.permission.WAKE_LOCK" />

    <!-- Creates a custom permission so only this app can receive its messages. -->
    <permission
        android:name="com.dorji.finalproject.permission.C2D_MESSAGE"
        android:protectionLevel="signature" />

    <uses-permission android:name="com.dorji.finalproject.permission.C2D_MESSAGE" />

    <!-- This app has permission to register and receive data message. -->
    <uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />

    <!-- Network State Permissions to detect Internet status -->
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

    <!-- Permission to vibrate -->
    <uses-permission android:name="android.permission.VIBRATE" />

    <!-- Main activity. -->
    <application
        android:debuggable="true" 
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <!-- Register Activity -->
        <activity
            android:name="com.dorji.finalproject.LoginLayout"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

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

        <!-- Main Activity -->
        <activity
            android:name="com.dorji.finalproject.MainActivity"
            android:configChanges="orientation|keyboardHidden"
            android:label="@string/app_name" >
        </activity>

        <receiver
            android:name="com.google.android.gcm.GCMBroadcastReceiver"
            android:permission="com.google.android.c2dm.permission.SEND" >
            <intent-filter>

                <!-- Receives the actual messages. -->
                <action android:name="com.google.android.c2dm.intent.RECEIVE" />
                <!-- Receives the registration id. -->
                <action android:name="com.google.android.c2dm.intent.REGISTRATION" />

                <category android:name="com.dorji.finalproject" />
            </intent-filter>
        </receiver>

        <service android:name="com.dorji.finalproject.GCMIntentService" />
    </application>

</manifest>
4

2 に答える 2

1

このエラーは、Google の GCM ドキュメントで次のように定義されています。

public static final 文字列 ERROR_SERVICE_NOT_AVAILABLE

デバイスが応答を読み取ることができないか、後で再試行できるサーバーからの 500/503 がありました。アプリケーションは、指数バックオフを使用して再試行する必要があります。

一時的なエラーの可能性があるようです。あと数回再試行して、何が起こるかを確認します。

また、@ol_v_er は、GCM docsに記載されているように、デバイスが GCM 要件を満たしていることを確認する必要があると述べました。

  • Android 2.2 以降を実行し、Google Play ストア アプリケーションもインストールされているデバイス、または Google API を使用して Android 2.2 を実行するエミュレータが必要です。ただし、Google Play ストアを通じて Android アプリケーションを展開することに限定されません。
  • Google サービスの既存の接続を使用します。3.0 より前のデバイスの場合、ユーザーはモバイル デバイスで Google アカウントを設定する必要があります。Android 4.0.4 以降を実行しているデバイスでは、Google アカウントは必須ではありません。
于 2013-03-21T14:38:09.307 に答える
0

このメッセージ SERVICE_NOT_AVAILABLE が理由もなく表示され、stackoverflow の回答で問題が解決されない場合は、必要なのは忍耐だけである可能性があるため、読み進めてください。だからセルシンの答えは私のために働いた。

私のアプリが nexus で正常に動作した後、突然 SERVICE_NOT_AVAILABLE メッセージの生成が開始され、登録 ID が null として表示されました。次に、2 つの異なるエミュレーターでアプリを試してみましたが、どちらでも問題なく動作しました。

さまざまなスタックオーバーフローの投稿ですべての提案を試しましたが、役に立ちませんでした。私は一晩中デバイスでアプリを実行したままにし、画面の半分を SERVICE_NOT_AVAILABLE メッセージで埋め尽くしました。

アプリをアンインストールし、デバイスの電源を切り、就寝しました。翌朝、インストールされている nexus の電源を入れ、アプリを再度実行したところ、完全に機能しました。

奇妙なことは、デバイスが新しいものとして GCM サーバーに登録されたことです。私のアプリケーションサーバーは、追加された別のデバイスもノッチアップしました。残念ながら、私のアプリケーション サーバーは最低限のものであり、持続性やログ記録がないため、これ以上調査することはできません。

他の誰かがこの現象を経験しましたか?

于 2013-09-15T08:09:38.923 に答える