在某些设备上下班后收到 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 通知的主要内容,如果未能解决你的问题,请参考以下文章

某些设备在 android 中未收到推送通知

GCM 通知结果=[错误=未注册]

GCM - 多个通知

GCM 通知仅到达某些设备

当应用程序未运行时,GCM 推送通知未在某些设备中显示

如何接收应用程序停止或设备关闭时发送的 GCM 推送通知?