某些设备在几小时后收到Gcm通知

时间:2015-07-27 14:25:37

标签: google-cloud-messaging

我正在尝试使用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

或单独提及接收者和注册意图过滤器

<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-请仔细阅读整个问题,然后发表您的意见或答案,或在必要时标记副本。

3 个答案:

答案 0 :(得分:7)

  • GCM通过Google Play服务工作
  • 设备通过端口5228上的TCP
  • 连接到Google Play服务
  • 如果端口5228被阻止,设备应使用端口443作为后备,但显然他们有时不会使用后备(发生在我的多个设备上)
  • 设备会在移动设备上每隔28分钟向Google Play服务发送心跳包,在无线网络上发送15分钟
  • 您可以检查连接状态,心跳间隔,连接地址和端口等。在设备上拨打*#*#426#*#*

当设备无法通过wifi连接到Google Play服务或经常断开连接时,如何解决常见问题

  1. 打开端口5228
  2. 配置您的路由器,以便在等待至少15分钟之前不会终止非活动的tcp连接
  3. 有关详细信息,请参阅this post on Google Product Forums

答案 1 :(得分:1)

我遇到了同样的问题。

首先,确保您的消息不超过最大字符长度。 其次,很多人告诉我这可能是来自手机或网络的防火墙问题,但其他应用程序(如whatsapp)运行顺畅。 几个星期后,它停止了延误,我的问题一个人就解决了。

我知道这不是一个真正的解决方案,但请等待,开发其他东西,然后在几周内再次测试它。

答案 2 :(得分:1)

由于您按预期在某些设备上收到通知,因此我假设您发布的清单和Java代码基本上应该没问题。你可以发布你的GcmBroadcastReceiver课吗?它应该是这样的:

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。