应用未打开时不发送 GCM 消息?

Posted

技术标签:

【中文标题】应用未打开时不发送 GCM 消息?【英文标题】:GCM Message is not sent when the app is not open? 【发布时间】:2014-04-28 06:26:39 【问题描述】:

最近我在我的应用上实现了 GCM。我按照这个网站上的教程代码进行操作

http://javapapers.com/android/google-cloud-messaging-gcm-for-android-and-push-notifications/

但是,行为很奇怪,例如

如果我杀死应用程序,那么当我发送通知时,它不会立即显示在通知栏上,而是只有当用户再次打开应用程序时,例如,如果我在杀死应用程序时发送两个通知,我会收到只有当我打开应用程序时才会有两个通知。

这是 GCM 的行为吗?正如我所期望的,它应该类似于whatsapp(即使我没有打开应用程序,设备仍应收到通知并显示)

这是我的代码。感谢您的帮助

GcmBroadcastReceiver

public class GcmBroadcastReceiver extends WakefulBroadcastReceiver 

    @Override
    public void onReceive(Context context, Intent intent) 
        ComponentName comp = new ComponentName(context.getPackageName(),
                GCMNotificationIntentService.class.getName());
        startWakefulService(context, (intent.setComponent(comp)));
        setResultCode(Activity.RESULT_OK);
    

GCMNotificationIntentService

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)) 

                for (int i = 0; i < 3; i++) 
                    Log.i(TAG, "Working... " + (i + 1) + "/5 @ " + SystemClock.elapsedRealtime());
                    try 
                        Thread.sleep(5000);
                     catch (InterruptedException e) 
                    
                

                Log.i(TAG, "Completed work @ " + SystemClock.elapsedRealtime());
                sendNotification(getResources().getString(R.string.gcm_news_remind));
                Log.i(TAG, "Received: " + extras.toString());
            
        
        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, SplashScreen.class), 0);

        NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(
                this).setSmallIcon(R.drawable.ic_launcher)
                .setContentTitle(getResources().getString(R.string.app_name))
                .setStyle(new NotificationCompat.BigTextStyle().bigText(msg))
                .setContentText(msg);

        mBuilder.setContentIntent(contentIntent);
        mNotificationManager.notify(NOTIFICATION_ID, mBuilder.build());
        Log.d(TAG, "Notification sent successfully.");
    

清单

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.oshpedia"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.GET_ACCOUNTS" />
    <uses-permission android:name="android.permission.WAKE_LOCK" />
    <uses-permission android:name="com.example.oshpedia.permission.C2D_MESSAGE" />
    <uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.VIBRATE" />

    <permission
        android:name="com.example.oshpedia.permission.C2D_MESSAGE"
        android:protectionLevel="signature" />

    <uses-sdk
        android:minSdkVersion="9"
        android:targetSdkVersion="19" />

    <application
        android:name=".Defination.MyApp"
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:largeHeap="true"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".Activity.SplashScreen"
            android:label="@string/app_name"
            android:theme="@android:style/Theme.NoTitleBar.Fullscreen" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name=".Activity.Main"
            android:screenOrientation="portrait"
            android:windowSoftInputMode="stateHidden|adjustPan" >
        </activity>
        <activity
            android:name=".Activity.AboutUs"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.AboutApp"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.Disclaimer"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.EnquiryForm"
            android:screenOrientation="portrait"
            android:windowSoftInputMode="stateHidden|adjustPan" >
        </activity>
        <activity
            android:name=".Activity.NewsDetail"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.EnquiryDetail"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.SearchResult"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.LawDetail"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.AdvanceSearch"
            android:screenOrientation="portrait" >
        </activity>
        <activity
            android:name=".Activity.VideoChannel"
            android:screenOrientation="portrait" >
        </activity>
        <activity android:name=".Activity.VideoViewer" >
        </activity>

        <receiver
            android:name=".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.example.oshpedia" />
            </intent-filter>
        </receiver>

        <service android:name=".GCM.GCMNotificationIntentService" />
    </application>

</manifest>

【问题讨论】:

【参考方案1】:

WhatsApp 并不完全依赖 GCM(我不确定它是否完全依赖 GCM)。如果您打开 Android 设备的设置并转到应用程序,然后转到运行服务(如果您使用的 Android 版本与我使用的不同,菜单可能会有所不同,即 2.3.6),您会看到有一个WhatsApp 服务在后台运行(假设您已安装 WhatsApp)。如果您停止该服务,您将停止接收来自 WhatsApp 的消息,直到您再次启动该应用程序。

您的应用只能接受 GCM 广播,前提是您没有明确终止它。一旦你杀死它,它不会收到任何广播,直到你下次启动它(根据我读到的,从 Android 3.x 开始就是这种情况)。但是,如果您将应用移至后台(例如,通过启动另一个应用或点击返回按钮直到您移至主屏幕或另一个应用),您的应用仍会从 GCM 获得消息。

【讨论】:

【参考方案2】:

GCM 引擎将在检测到您与它们建立连接后尝试传递消息。我猜当你杀死你的应用程序时,等待获取消息的开放套接字也会被杀死,并且他们不会再次尝试传递消息。

当您再次打开应用程序时,所有待处理的消息都会被传递,因为GCM 引擎会在检测到您已与它们建立连接后尝试传递所有这些消息,这就是您在同一时间。

【讨论】:

这是否意味着这是标准行为?如何将代码更改为whatsapp之类的通知? (即使应用程序没有启动,仍然可以显示收到的通知)?谢谢 是的,我会说这是预期的行为。为了让它即使在应用程序不在前台时也能收到通知,您可能必须更改代码以使您的IntentService 处理与GCM 部分(这是后台服务)的连接。然而,如果 android 操作系统需要释放内存并且这个IntentService 被选择停止,这将有一些风险被杀死。这可能会对您有所帮助:developer.android.com/training/run-background-service/…

以上是关于应用未打开时不发送 GCM 消息?的主要内容,如果未能解决你的问题,请参考以下文章

GCM Phonegap,如何将消息发送到手机的核心通知

一旦来自附近位置的新数据进入relatime数据库,如何发送通知(当应用程序未打开时)?

不保证 Gcm 交付

GCM 中使用上游消息传递的消息流程图

GCM:如何从服务器向应用程序发送消息(丢失注册 ID、通知 ID ...)

GCM 多次推送同一消息 (PushSharp)