在某些设备上下班后收到 Gcm 通知
Posted
技术标签:
【中文标题】在某些设备上下班后收到 Gcm 通知【英文标题】:Gcm notification received after hours on some devices 【发布时间】:2015-07-27 14:25:37 【问题描述】:我正在尝试使用 gcm 通知。我的服务器代码运行良好,我得到了确认。
问题是通知发送正确:
1) 在大多数设备中,通知是即时收到的。 在 google nexus、sony 手机上测试。
2) 其他设备也在几个小时后收到通知。是的,小时。在 Karbonn、Micromax 一些手机上进行了测试。
注意:
所有设备都连接到同一个 wifi,因此网络连接不是问题。在服务器端使用 php; 关于这个主题有几个未解决的问题。我在此列出其中的一些:
gcm notification is not working on some devices like micromax
One device doesn't receive push notifications (GCM)
Push notifications delay with GCM
其他有类似问题的人也附上您的问题。
整改失败:
在解决了几个问题后,我还对代码进行了更改,开发人员在这些问题中找到了他们的解决方案,例如
从onHandleIntent()
中删除这行代码
GcmBroadcastReceiver.completeWakefulIntent(intent);
或在服务器代码中将 delay_while_ideal
值更改为 true/false
。
或者单独提到receiver和registration-intent-filter
<intent-filter>
<action android:name="com.google.android.c2dm.intent.RECEIVE" />
<category android:name="com.nothing.gcmclient" />
</intent-filter>
<intent-filter>
<action android:name="com.google.android.c2dm.intent.REGISTRATION" />
<category android:name="com.nothing.gcmclient" />
</intent-filter>
代码: Android.manifest
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.nothing.gcmclient"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="9"
android:targetSdkVersion="22" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<permission
android:name="com.nothing.gcmclient.permission.C2D_MESSAGE"
android:protectionLevel="signature" />
<uses-permission android:name="com.nothing.gcmclient.permission.C2D_MESSAGE" />
<uses-permission android:name="com.nothing.gcmclient.c2dm.permission.RECEIVE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="ANDROID.PERMISSION.WRITE_EXTERNAL_STORAGE"></uses-permission>
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<activity
android:name=".RegisterActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".MainActivity"
android:configChanges="orientation|keyboardHidden"
android:label="@string/app_name" >
</activity>
<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="com.nothing.gcmclient" />
</intent-filter>
<intent-filter>
<action android:name="com.google.android.c2dm.intent.REGISTRATION" />
<category android:name="com.nothing.gcmclient" />
</intent-filter>
</receiver>
<service android:name=".GCMNotificationIntentService"></service>
<activity
android:name=".ChatActivity"
android:label="@string/title_activity_chat" >
</activity>
<activity
android:name=".RegisterScreen"
android:label="@string/title_activity_register_screen" >
</activity>
<activity
android:name=".RegisterChatButtonActivity"
android:label="@string/title_activity_register_chat_button" >
</activity>
<activity
android:name=".ChatHistory"
android:label="@string/title_activity_chat_history" >
</activity>
<activity
android:name=".MessageScreen"
android:label="@string/title_activity_message_screen" >
</activity>
</application>
</manifest>
代码:GCMNotificationIntentService.java
public class GCMNotificationIntentService extends IntentService
public static final int NOTIFICATION_ID = 1;
private NotificationManager mNotificationManager;
NotificationCompat.Builder builder;
public GCMNotificationIntentService()
super("GcmIntentService");
public static final String TAG = "GCMNotificationIntentService";
@Override
protected void onHandleIntent(Intent intent)
Bundle extras = intent.getExtras();
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());
else if (GoogleCloudMessaging.MESSAGE_TYPE_MESSAGE
.equals(messageType))
String sender=extras.get(Config.SENDER).toString().toLowerCase();
String message=extras.get(Config.MESSAGE_KEY).toString();
if(!RegisterActivity.appVisible==true)
sendNotification("New message Received from "+ extras.get(Config.SENDER));
//GcmBroadcastReceiver.completeWakefulIntent(intent);
private void sendNotification(String msg)
Log.d(TAG, "Preparing to send notification...: " + msg);
mNotificationManager = (NotificationManager) this
.getSystemService(Context.NOTIFICATION_SERVICE);
PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
new Intent(this, ChatHistory.class), 0);
NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(
this).setSmallIcon(R.drawable.gcm_cloud)
.setContentTitle("New Notification")
.setStyle(new NotificationCompat.BigTextStyle().bigText(msg))
.setContentText(msg);
mBuilder.setContentIntent(contentIntent);
mNotificationManager.notify(NOTIFICATION_ID, mBuilder.build());
Log.d(TAG, "Notification sent successfully.");
请专家调查问题并提出适当的原因和解决方案。如果我的应用程序仅适用于某些手机,那么它将毫无用处。如果您需要我们应用程序中的更多文件,请告知。
PS- 请阅读整个问题,然后发布您的 cmets 或答案,或在必要时标记重复。
【问题讨论】:
听起来像是设备问题或 GCM 服务器问题(可能是设备)。这两个你都无法修复 在 7 台设备上试用过。三连败。该百分比足以相信设备特定的问题 你可以试试here和here这两个官方demo app。如果问题仍然存在,我认为这肯定是特定于设备的问题,所以也许你可以找到一些公司的支持公司。 您必须正确处理gcm注册id,实际上有时gcm识别gcm注册id是非常突然的情况。您的整个代码很好,您只需正确处理 gcm id 并每次更新。 @user2653926 抱歉,正确的 USSD 代码是*#*#426#*#*
,这里是谷歌论坛上相应帖子的链接:productforums.google.com/forum/#!msg/nexus/fslYqYrULto/…
【参考方案1】:
GCM 通过 Google Play 服务工作
设备通过 TCP 端口 5228 连接到 Google Play 服务
如果端口 5228 被阻止,设备应使用端口 443 作为后备,但显然它们有时只是不使用后备(发生在我的多个设备上)
设备在移动设备上每 28 分钟向 Google Play 服务发送一个心跳包,在 wifi 上每 15 分钟发送一次
在设备上拨打*#*#426#*#*
可以查看连接状态、心跳间隔、连接地址和端口等
如何解决设备无法通过 wifi 连接到 Google Play 服务或频繁断开连接时的常见问题:
-
开放5228端口
配置您的路由器,使其在等待至少 15 分钟之前不会终止非活动的 tcp 连接
更多信息请参见this post on Google Product Forums。
【讨论】:
感谢您的信息。该代码特别有用。 你能解释一下如何“打开5228端口”吗?【参考方案2】:我遇到了同样的问题。
首先,确保您的消息不超过最大字符长度。 其次,很多人告诉我这可能是手机或网络的防火墙问题,但其他应用程序(如whatsapp)运行良好。 几周后,它停止了延误,我的问题独自解决了。
我知道这不是一个真正的解决方案,但请尝试等待,开发其他东西并在几周后再次测试它。
【讨论】:
不,就我而言,我已经放弃支持所有设备并继续构建应用程序。当应用程序完成后,我刚刚重新测试并且一切正常......我的建议是不要就此止步,完成你的应用程序,就好像它工作正常,而不是重新测试。可能 Google 系统具有一定的应用可靠性或某种可扩展的评分系统【参考方案3】:由于您按预期在某些设备上收到通知,我假设您发布的清单和 Java 代码基本上应该没问题。你能发布你的GcmBroadcastReceiver
-class 吗?应该是这样的:
public class GcmBroadcastReceiver extends WakefulBroadcastReceiver
/**
* Starts GcmIntentService which handles the intent.
*/
@Override
public void onReceive(Context context, Intent intent)
// Explicitly specify that GcmIntentService will handle the intent.
ComponentName comp = new ComponentName(context.getPackageName(), GCMNotificationIntentService.class.getName());
// Start the service, keeping the device awake while it is launching.
startWakefulService(context, (intent.setComponent(comp)));
setResultCode(Activity.RESULT_OK);
这个类负责唤醒你的应用程序(如果你的应用程序的进程还没有运行)并调用你的服务,如果你没有像这样实现它,Android系统将不会启动你的 GCMNotificationIntentService 因为应用程序没有正在运行,因此无法显示通知。这也可以解释为什么某些设备会立即收到通知。所以我假设您在每台设备上都会收到通知,但不会立即显示。
此外,如果您的 BroadcastReceiver 是正确的,则绝对没有必要在 GCMNotificationIntentService 中调用GcmBroadcastReceiver.completeWakefulIntent(intent);
,因为这条线甚至没有任何效果,因为它从未被调用过(如果我的假设是正确的,那么您的 BroadcastReceiver 就是问题所在)。
据我所知,您使用的是旧版本的 GCM-API。我不知道您是否知道几周后可以使用较新的 GCM-API。前段时间我也遇到了与旧版本类似的问题,但由于我使用的是新 API(包括替换旧注册 ID 的 InstanceID-API),我再也没有遇到任何通知延迟问题。
我强烈建议实施新的 GCM-API。请参阅docs for further information。 that´s the doc 用于 InstanceID-API。
【讨论】:
会经历那个。谢谢以上是关于在某些设备上下班后收到 Gcm 通知的主要内容,如果未能解决你的问题,请参考以下文章