在Android平台推送通知

我正在寻找一个应用程序接收来自服务器的推送警报。 我发现了几个方法来做到这一点。

  1. 短信 – 拦截收到的短信,并从服务器发起一个拉
  2. 定期轮询服务器

每个都有其自身的局限性。 短信 – 无法保证到达时间。 投票可能会耗尽电池。

你有更好的建议吗? 非常感谢。

Google的正式答案是Android云到设备消息传递框架 (已弃用) Google云消息传递 (已弃用) Firebase云消息传递

它将在Android> = 2.2(在具有Play商店的手机上)工作。

从我给类似问题的答案交叉发布 – Android支持接近实时推送通知?

我最近开始使用Android的MQTT http://mqtt.org作为这种事情的一种方式(即推送通知不是SMS,而是数据驱动,几乎是即时消息传递,而不是轮询等);

我有一个博客文章的背景资料,以防万一它有帮助

http://dalelane.co.uk/blog/?p=938

(注意:MQTT是一项IBM技术,我应该指出,我为IBM工作。)

Android云到设备消息传递框架

重要提示:截至2012年6月26日,C2DM已被正式废弃。这意味着C2DM已经停止接受新用户和配额请求。 没有新的功能将被添加到C2DM。 但是,使用C2DM的应用程序将继续工作。 鼓励现有的C2DM开发人员迁移到称为Google Cloud Messaging for Android(GCM)的新版C2DM。 有关更多信息,请参阅C2DM-GCM迁移文档。 开发人员必须使用GCM进行新的开发。

请检查以下链接:

http://developer.android.com/guide/google/gcm/index.html

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

  1. 在Google Cloud上创建/注册应用程序
  2. 使用开发安装Cloud SDK
  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=“<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(推送通知的设备令牌)

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

添加常量和类变量

 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”,所以让我们用这种方式创建这个类更新接收者类代码

 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”,所以让我们用这种方式创建这个类更新接收者类代码

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

我对Android推送通知的理解/体验是:

  1. C2DM GCM – 如果你的目标Android平台是2.2+,那么去吧。 只有一个问题,设备用户必须始终使用Google帐户登录以获取消息。

  2. MQTT – 基于Pub / Sub的方法需要来自设备的有效连接,如果没有合理实施,可能会耗尽电池。

  3. 执事 – 由于社区支持有限,从长远来看可能不会很好。

编辑 :2013年11月25日添加

GCM – Google说…

对于3.0之前的设备,这需要用户在他们的移动设备上设置他们的Google帐户。 运行Android 4.0.4或更高版本的设备不需要Google帐户。*

在Meteor网络服务器上开发一个基于Android的推送通知的Java库是一个新的开源工作。 您可以在Deacon Project博客中查看它,在这里您可以找到Meteor和项目的GitHub存储库的链接。 我们需要开发者,所以请传播这个词!

您可以使用Xtify( http://developer.xtify.com ) – 他们有一个推送通知web服务,与他们的SDK一起工作。 它是免费的,到目前为止,这对我来说真的很好。

要么….

3)保持与服务器的连接,每隔几分钟发送一次保持连接,服务器可以立即发送消息。 这就是Gmail,Google Talk等的工作原理。

我建议使用GCM – Google Cloud Messaging for Android它是免费的,对于简单的用途应该是非常简单的。

但是,它需要维护第三方服务器代表您发送通知。 如果你想避免有一些非常好的Android推送通知服务工业解决方案:

  • 城市飞艇 – 每月最多可免费收到1M通知,之后每1000通知您收取一次费用
  • PushApps – 每月免费发送1M通知,每月19.99次免费通知
  • PushWoosh – 1M设备免费,优惠计划从39欧元

Diclaimer – 我在PushApps工作,并在我的应用程序中使用他们的产品一年多了。

截至2016年5月18日Firebase是Google的移动开发者统一平台,包括推送通知。

恐怕你已经找到了两种可能的方法。 谷歌,至少在最初,将实施一个GChat API,你可以使用推/拉实施。 不幸的是,这个库被Android 1.0切割了。

我不知道这是否仍然有用。 我用http://www.pushlets.com/上的java库获得了类似的结果;

Althoug在服务中执行它并不会阻止android将其关闭,从而导致对listener线程的查杀。

Google C2DM现在已经贬值了,为此,您可以使用新的服务GCM(Google Cloud Messaging)。 有关文档,请参阅http://developer.android.com/guide/google/gcm/gs.html

C2DM:您的应用程序用户必须拥有Gmail帐户。

MQTT:当你的连接达到1024时,它会停止工作,因为它使用了linux的“select model”。

有一个免费的推送服务和Android的API,你可以尝试一下: http : //push-notification.org

免费和简单的方法:

如果您的目标用户群不是很大(小于1000),并且您想要免费服务,那么Airbop是最好的,也是最方便的。

Airbop网站它通过API使用Google云消息传递服务,并提供良好的性能。 我已经将它用于我的两个项目,并且很容易实现它。

像和Urbanship服务是优秀的,但提供了一个完整的部署堆栈,而不仅仅是推送通知的事情。

如果只是推送服务是你的目标,Airbop将正常工作。

我没有使用Pushwoosh ,但也是一个很好的选择。 它允许免费推送到1,000,000个设备

我会建议使用SMS和HTTP。 如果用户没有登录发送短信通知他们有一个消息等待。

这就是爱立信实验室服务的工作原理: https : //labs.ericsson.com/apis/mobile-java-push/

如果你自己实现这个棘手的部分是删除传入的短信,用户看不到它。 或者,如果他们看到你的情况,也许没关系。

看起来像这样: 使用BroadCastReceiver删除SMS – Android

是的,编写这样的代码可能是危险的,你可能会毁了一个人的生活,因为你的应用程序删除了它不应该有的短信。

您可以使用Google云消息传递或GCM ,它可以免费且易于使用。 你也可以使用PushWoosh这样的第三方推送服务器,这给你更多的灵活性

有许多像Urban Airship ,Xtify, Mainline这样的第三方服务器,它们不仅允许在Android上发送,还可以在iOs,Windows Phone上发送。