28

更新:以下のコードの問題を修正したので、これは使用方法の優れた基本的な実例になりますGCM


だから、私はGCM自分のアプリにAndroidを実装しようとしています。マニフェストに追加した関連部分は次のとおりです。

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

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

..。

<receiver
    android:name="com.google.android.gcm.GCMBroadcastReceiver"
    android:permission="com.google.android.c2dm.permission.SEND" >
    <intent-filter>
        <action android:name="com.google.android.c2dm.intent.RECEIVE" />
        <action android:name="com.google.android.c2dm.intent.REGISTRATION" />
        <category android:name="com.badbob.app.gmctestapp" />
    </intent-filter>
</receiver>

<service android:name=".GCMIntentService" />

メインアクティビティのonCreateに次のコードを追加しました。

    GCMRegistrar.checkDevice( this );
    GCMRegistrar.checkManifest( this );
    final String regId = GCMRegistrar.getRegistrationId( this );
    if( regId.equals( "" ) ) {
        GCMRegistrar.register( this, GCM_SENDER_ID );
    }
    else {
        Log.v( LOG_TAG, "Already registered" );
    }

私も次のGCMIntenetServiceようなクラスを作成しました:

public class GCMIntentService extends GCMBaseIntentService {

    private static final String LOG_TAG = "GetAClue::GCMIntentService";

    public GCMIntentService() {
        super( GCM_SENDER_ID );
        // TODO Auto-generated constructor stub
        Log.i( LOG_TAG, "GCMIntentService constructor called" );
    }

    @Override
    protected void onError( Context arg0, String errorId ) {
        // TODO Auto-generated method stub
        Log.i( LOG_TAG, "GCMIntentService onError called: " + errorId );
    }

    @Override
    protected void onMessage( Context arg0, Intent intent ) {
        // TODO Auto-generated method stub
        Log.i( LOG_TAG, "GCMIntentService onMessage called" );
        Log.i( LOG_TAG, "Message is: " + intent.getStringExtra( "message" ) );
    }

    @Override
    protected void onRegistered( Context arg0, String registrationId ) {
        // TODO Auto-generated method stub
        Log.i( LOG_TAG, "GCMIntentService onRegistered called" );
        Log.i( LOG_TAG, "Registration id is: " + registrationId );
    }

    @Override
    protected void onUnregistered( Context arg0, String registrationId ) {
        // TODO Auto-generated method stub
        Log.i( LOG_TAG, "GCMIntentService onUnregistered called" );
        Log.i( LOG_TAG, "Registration id is: " + registrationId );
    }
}

これを実行すると、 LogCatでこれを取得します。

07-11 11:28:46.340: V/GCMRegistrar(27435): Registering receiver
07-11 11:28:46.370: D/GCMRegistrar(27435): resetting backoff for com.badbob.app.getacluebeta
07-11 11:28:46.380: V/GCMRegistrar(27435): Registering app com.badbob.app.getacluebeta of senders 128205395388

他の投稿から収集したものから、登録IDを取得する必要がありますが、そうでLogCatはありません。またonRegistered()GCMIntentService呼び出されることはありません。では、何が間違っているのでしょうか。

4

9 に答える 9

18

これは正しくありません

protected GCMIntentService( String senderId ) {         
super(senderId);}

ドキュメントに記載されているとおり。GCMIntentServiceのPUBLIC、NOARGUMENTコンストラクターを宣言する必要があります。そうしないと、GCMIntentServiceをバックグラウンドインテントサービスによって適切にインスタンス化できません。

public GCMIntentService(){
super(senderID);}

senderIDは、新しいGCMで変更されなくなるため、ハードコードされた定数文字列にすることができます。正しいsenderIDを使用することも非常に重要です。24時間はあなたがアクティブになるのに十分な長さなので、上記の解決策が機能しない場合は、間違ったsenderIDを使用しています。他のすべては素晴らしく見えます。

送信者IDは、GoogleAPIアクセスページを閲覧しているときにWebブラウザのURLに含まれています。GCMをクリックすると、ブラウザのURLに12桁の数字が表示されます。それが使用する正しいキーです。APIキーではありません。これは、アプリサーバー側で使用されます。

于 2012-07-11T18:47:03.933 に答える
8

私も今までこの厄介なバグを抱えていました。問題は、別のパッケージ内でintentserviceを宣言したことです... androidマニフェストで名前を宣言し、classnotfound例外が発生していなくても...エラーはまったく見つかりませんでした... intentserviceはルートパッケージに含まれています。

于 2012-12-27T10:14:00.647 に答える
3

コードに問題があると思います。

マニフェストファイルで独自のブロードキャストレシーバーの登録を行う必要があります。そうすると、そのレシーバーがをトリガーし<service android:name=".GCMIntentService" />ます。

したがって、あなたは私が以下に書くようなことをしなければなりません。

  • 受信者は次のように宣言する必要があります。

    <receiver
        android:name="your.package.name.YourBroadCastReceiver"
        android:permission="com.google.android.c2dm.permission.SEND" >
        <intent-filter>
            <action android:name="com.google.android.c2dm.intent.RECEIVE" />
            <action android:name="com.google.android.c2dm.intent.REGISTRATION" />
        </intent-filter>
    

  • サービスを開始する放送受信機。

    public class YourBroadCastReceiver extends BroadcastReceiver {
    
    @Override
     public final void onReceive(Context context, Intent intent) {
         GCMIntentService .runIntentInService(context, intent);
         setResult(Activity.RESULT_OK, null, null);
     }  
    }
    

良い例を見つけることができる公式のGCMドキュメントを参照することをお勧めします。

そして...メインのGoogleAPIコンソールページでGoogleクラウドメッセージングサービスを有効にすることを忘れないでください。

それが役立つかどうか教えてください!

于 2012-07-11T19:51:41.597 に答える
3

GCMについては以下のチュートリアルを参照してください:-

http://www.androidhive.info/2012/10/android-push-notifications-using-google-cloud-messaging-gcm-php-and-mysql/

于 2013-08-05T10:23:28.997 に答える
2

ここでは、RegIDと通知を最初から取得する方法のいくつかの手順を記述しました

  1. GoogleCloudでアプリを作成/登録する
  2. 開発を伴うCloudSDKのセットアップ
  3. GCM用にプロジェクトを構成する
  4. デバイス登録IDを取得する
  5. プッシュ通知を送信する
  6. プッシュ通知を受信する

完全なチュートリアルは以下のURLリンクにあります

Androidプッシュ通知入門:最新のGoogleクラウドメッセージング(GCM)-ステップバイステップの完全なチュートリアル

ここに画像の説明を入力してください

登録ID(プッシュ通知用のデバイストークン)を取得するためのコードスニップ。

GCM用にプロジェクトを構成する


AndroidManifestファイルを更新します

プロジェクトでGCMを有効にするには、マニフェストファイルにいくつかの権限を追加する必要があります。AndroidManifest.xmlに移動し、以下のコードを追加します。権限の追加

<uses-permission android:name="android.permission.INTERNET”/>
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.WAKE_LOCK" />

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

<uses-permission android:name=“.permission.RECEIVE" />
<uses-permission android:name=“&lt;your_package_name_here>.permission.C2D_MESSAGE" />
<permission android:name=“&lt;your_package_name_here>.permission.C2D_MESSAGE"
        android:protectionLevel="signature" />

GCMブロードキャストレシーバー宣言を追加

アプリケーションタグにGCMブロードキャストレシーバー宣言を追加します

<application
        <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="" />
            </intent-filter]]>

        </receiver]]>
     
<application/>

GCMセルビア宣言を追加

<application
     <service android:name=".GcmIntentService" />
<application/>

登録ID(プッシュ通知用のデバイストークン)を取得します

次に、起動/スプラッシュアクティビティに移動します

定数とクラス変数を追加する

private final static int PLAY_SERVICES_RESOLUTION_REQUEST = 9000;
public static final String EXTRA_MESSAGE = "message";
public static final String PROPERTY_REG_ID = "registration_id";
private static final String PROPERTY_APP_VERSION = "appVersion";
private final static String TAG = "LaunchActivity";
protected String SENDER_ID = "Your_sender_id";
private GoogleCloudMessaging gcm =null;
private String regid = null;
private Context context= null;

OnCreateメソッドとOnResumeメソッドを更新します

@Override
protected void onCreate(Bundle savedInstanceState)
{
     super.onCreate(savedInstanceState);
     setContentView(R.layout.activity_launch);
     context = getApplicationContext();
         if (checkPlayServices()) 
     {
            gcm = GoogleCloudMessaging.getInstance(this);
            regid = getRegistrationId(context);

            if (regid.isEmpty())
            {
                registerInBackground();
            }
            else
            {
            Log.d(TAG, "No valid Google Play Services APK found.");
            }
      }
 }

@Override protected void onResume()
{
       super.onResume();       checkPlayServices();
}


# Implement GCM Required methods (Add below methods in LaunchActivity)

private boolean checkPlayServices() {
        int resultCode = GooglePlayServicesUtil.isGooglePlayServicesAvailable(this);
        if (resultCode != ConnectionResult.SUCCESS) {
            if (GooglePlayServicesUtil.isUserRecoverableError(resultCode)) {
                GooglePlayServicesUtil.getErrorDialog(resultCode, this,
                        PLAY_SERVICES_RESOLUTION_REQUEST).show();
            } else {
                Log.d(TAG, "This device is not supported - Google Play Services.");
                finish();
            }
            return false;
        }
        return true;
 }

private String getRegistrationId(Context context) 
{
   final SharedPreferences prefs = getGCMPreferences(context);
   String registrationId = prefs.getString(PROPERTY_REG_ID, "");
   if (registrationId.isEmpty()) {
       Log.d(TAG, "Registration ID not found.");
       return "";
   }
   int registeredVersion = prefs.getInt(PROPERTY_APP_VERSION, Integer.MIN_VALUE);
   int currentVersion = getAppVersion(context);
   if (registeredVersion != currentVersion) {
        Log.d(TAG, "App version changed.");
        return "";
    }
    return registrationId;
}

private SharedPreferences getGCMPreferences(Context context) 
{
    return getSharedPreferences(LaunchActivity.class.getSimpleName(),
                Context.MODE_PRIVATE);
}

private static int getAppVersion(Context context) 
{
     try 
     {
         PackageInfo packageInfo = context.getPackageManager()
                    .getPackageInfo(context.getPackageName(), 0);
            return packageInfo.versionCode;
      } 
      catch (NameNotFoundException e) 
      {
            throw new RuntimeException("Could not get package name: " + e);
      }
}


private void registerInBackground() 
{     new AsyncTask() {
     Override
     protected Object doInBackground(Object... params) 
     {
          String msg = "";
          try 
          {
               if (gcm == null) 
               {
                        gcm = GoogleCloudMessaging.getInstance(context);
               }
               regid = gcm.register(SENDER_ID);               Log.d(TAG, "########################################");
               Log.d(TAG, "Current Device's Registration ID is: "+msg);     
          } 
          catch (IOException ex) 
          {
              msg = "Error :" + ex.getMessage();
          }
          return null;
     }     protected void onPostExecute(Object result) 
     { //to do here };
  }.execute(null, null, null);
}

:REGISTRATION_KEYを保存してください。PNメッセージをGCMに送信するために重要です。また、これを使用すると、GCMのみがプッシュ通知を送信するため、これはすべてのデバイスで一意になります。

プッシュ通知を受信する

GCMブロードキャストレシーバークラスを追加する

マニフェストファイルで「GcmBroadcastReceiver.java」をすでに宣言しているので、このクラス更新レシーバークラスコードをこのように作成しましょう

public class GcmBroadcastReceiver extends WakefulBroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) 
    {        ComponentName comp = new ComponentName(context.getPackageName(),
                GcmIntentService.class.getName());        startWakefulService(context, (intent.setComponent(comp)));
        setResultCode(Activity.RESULT_OK);
        Toast.makeText(context, “wow!! received new push notification", Toast.LENGTH_LONG).show();
    }
}

GCMサービスクラスを追加する

マニフェストファイルで「GcmBroadcastReceiver.java」をすでに宣言しているので、このクラス更新レシーバークラスコードをこのように作成しましょう

public class GcmIntentService extends IntentService
{     public static final int NOTIFICATION_ID = 1;     private NotificationManager mNotificationManager;     private final static String TAG = "GcmIntentService";     public GcmIntentService() {
     super("GcmIntentService");     
     }     @Override
     protected void onHandleIntent(Intent intent) {
          Bundle extras = intent.getExtras();
          Log.d(TAG, "Notification Data Json :" + extras.getString("message"));

          GoogleCloudMessaging gcm = GoogleCloudMessaging.getInstance(this);
          String messageType = gcm.getMessageType(intent);          if (!extras.isEmpty()) {          if (GoogleCloudMessaging.MESSAGE_TYPE_SEND_ERROR
               .equals(messageType)) {
               sendNotification("Send error: " + extras.toString());
          } else if (GoogleCloudMessaging.MESSAGE_TYPE_DELETED
          .equals(messageType)) {
          sendNotification("Deleted messages on server: "
          + extras.toString());          // If it's a regular GCM message, do some work.
          } else if (GoogleCloudMessaging.MESSAGE_TYPE_MESSAGE
          .equals(messageType)) {
          // This loop represents the service doing some work.
          for (int i = 0; i < 5; i++) {
               Log.d(TAG," Working... " + (i + 1) + "/5 @ "
               + SystemClock.elapsedRealtime());               try {
                    Thread.sleep(5000);
               } catch (InterruptedException e) {
               }
             }
             Log.i(TAG, "Completed work @ " + SystemClock.elapsedRealtime());
             sendNotification(extras.getString("message"));
           }
        }        // Release the wake lock provided by the WakefulBroadcastReceiver.
        GcmBroadcastReceiver.completeWakefulIntent(intent);
     }     // Put the message into a notification and post it.
     // This is just one simple example of what you might choose to do with
     // a GCM message.
     private void sendNotification(String msg) {          mNotificationManager = (NotificationManager) this
          .getSystemService(Context.NOTIFICATION_SERVICE);
          PendingIntent contentIntent = PendingIntent.getActivity(this, 0,          new Intent(this, LaunchActivity.class), 0);

          NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(          this)
          .setSmallIcon(R.drawable.icon)
          .setContentTitle("Ocutag Snap")
          .setStyle(new NotificationCompat.BigTextStyle().bigText(msg))
          .setContentText(msg)
          .setDefaults(Notification.DEFAULT_SOUND | Notification.DEFAULT_VIBRATE);

          mBuilder.setContentIntent(contentIntent);          mNotificationManager.notify(NOTIFICATION_ID, mBuilder.build());
     }
}
于 2013-11-29T06:43:26.213 に答える
0

これがあなたの特定の問題に関連しているかどうかはわかりませんが、私は同じ問題を経験していました。私の知る限り、すべてが正しくセットアップされていましたが、onRegistered()ハンドラーが呼び出されることはありませんでした。

GCMIntentServiceクラスの一部として、標準メソッドとサービスメソッドがonCreate()ありましonStart()た。onDestroy()それらを削除すると、期待どおりにハンドラーが呼び出されました。

私の想定では、これらのメソッドをオーバーライドすることで、必要なコードの初期化をバイパスしていました。

于 2012-07-11T21:45:17.240 に答える
0

ClouDeveloperの投稿を読んだ後、、、およびその他のonXXX()コールバックメソッドを削除onCreate()onDestroy()まし。また、私のGCMIntentService.onRegistered(Context, regId)はで呼び出されregIdます。

ClouDeveloperに感謝します!

于 2012-07-18T23:29:16.293 に答える
0

また、 GCMBaseIntentService.onRegistered()コールバックでアプリケーションサーバーに登録する必要があります。そうしないと、アプリメッセージを送信できなくなります。

GoogleのGCMデモクライアントは、正しいコードがすべて含まれた実用的なサンプルを提供しているので、実際にチェックアウトする必要があります。また、githubに動作するサンプルがあります:airbop-client。これは、GCM対応アプリに必要なアプリケーションサーバーとの相互作用が機能していることを考えると、もう少し完全な例を提供します。これは、最大1000台のデバイスを解放するAirBopサーバー(私が作成を手伝った)で動作します。

于 2013-01-06T17:32:57.393 に答える
0

登録から応答がない場合、主な理由の1つは、マニフェストファイルが正しく構成されていないことです。特に「package_name」(com.xxx.yyyなどのアプリパッケージ名)<permission><intent filter>正しく指定してください。

于 2013-06-28T22:46:11.907 に答える