Android的GCM基本实现

更新:我解决了下面的代码中的问题,所以这是一个很好的如何使用GCM基本工作示例


所以,我想实现Android GCM到我的应用程序。 以下是我已经添加到清单中的相关部分:

 <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" ); } 

我也创build了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 

从我从其他post收集到的信息,我应该在LogCat获得一个注册ID,但是我不会在GCMIntentService调用onRegistered() ,所以我做错了什么?

Related of "Android的GCM基本实现"

这是不正确的

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

正如它在文档中所述。 您必须为GCMIntentService声明一个PUBLIC,NO ARGUMENT构造函数。 否则GCMIntentService不能被背景意图服务正确地实例化。

 public GCMIntentService(){ super(senderID);} 

senderID可以是一个硬编码的常量string,因为它不会随着新的GCM而改变。 使用正确的senderID也是非常重要的。 如果我的上述解决scheme不起作用,24小时足够长,以致于您正在使用不正确的senderID。 其他一切看起来不错。

当您浏览Google API访问页面时,senderID位于您的Web浏览器的URL中。 点击GCM,一个12位数字将出现在浏览器的URL中。 这是使用正确的关键。 不是你的API密钥。 这在App服务器端使用。

我也有这个恼人的错误,直到现在。 问题是,我宣布在另一个包内的intentservice …即使我在android清单中声明的​​名称和没有classnotfoundexception已放置…没有任何错误可以find…所以我确保intentservice位于根包中。

我认为你在代码中有一些问题。

您应该在清单文件中注册您自己的广播接收器,并且该接收器将触发<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); } } 

我build议你看看官方的GCM 文档 ,你可以find一个很好的例子。

并且…不要忘记在主要的Google API控制台页面中启用Google云消息服务。

让我知道,如果有帮助!

在这里,我已经写了如何从头开始获取RegID和通知的几个步骤

  1. 在Google Cloud上创build/注册应用程序
  2. 使用开发安装Cloud SDK
  3. 为GCMconfiguration项目
  4. 获取设备注册ID
  5. 发送推送通知
  6. 接收推送通知

您可以在下面的URL链接中find完整的教程

Android推送通知入门:最新Google云消息传递(GCM) – 分步完整教程

在这里输入图像说明

代码片段获取注册ID(推送通知的设备令牌)。

为GCMconfiguration项目


更新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=“<your_package_name_here>.permission.C2D_MESSAGE" /> <permission android:name=“<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 Servie声明

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

获取注册ID(推送通知的设备令牌)

现在转到您的启动/飞溅活动

添加常量和类variables

 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广播接收机类

因为我们已经在我们的Manifest文件中声明了“GcmBroadcastReceiver.java”,所以让我们用这种方法创build这个类更新的接收者类代码

 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服务类

因为我们已经在我们的Manifest文件中声明了“GcmBroadcastReceiver.java”,所以让我们用这种方法创build这个类更新的接收者类代码

 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()); } } 

不知道这是否与您的具体问题有关,但是,我遇到了同样的问题。 一切都设置正确,据我所知,但onRegistered()处理程序从来没有被调用。

我有标准的onCreate()onStart()onDestroy()服务方法作为我的GCMIntentService类的一部分。 一旦我将他们删除,处理程序就像预期的那样被调用。

我的假设是通过重写这些方法,我绕过了所需的代码初始化。

在阅读ClouDeveloper的post后,我删除了onCreate()onDestroy()和其他onXXX()callback方法。 另外,我的GCMIntentService.onRegistered(Context, regId)是用regId调用的。

谢谢ClouDeveloper!

您还需要在GCMBaseIntentService.onRegistered()callback中向您的应用程序服务器进行注册。 否则,您将无法发送您的应用消息。

您应该真正结帐Google的GCM演示客户端,因为它提供了所有正确代码的工作示例。 我们也在github上有一个工作样本: airbop-client 。 它提供了一个稍微更完整的示例,因为它与应用程序服务器有工作交互,这对于启用GCM的应用程序是必需的。 它可以与AirBop服务器(我帮助创build)一起工作,最多可以有1000个设备。

如果没有得到注册响应,主要原因之一是你的清单文件没有正确configuration – 特别是在<permission><intent filter>给出了“package_name”(你的应用程序包名称像com.xxx.yyy) <intent filter>正确。