Firebase onMessageReceived 触发两次

Firebase onMessageReceived is triggering twice

我计划在 Firebase 服务收到消息时启动通知,但问题是通知触发两次(一次准时,第二次可能在 30 分钟后)这是我的代码:

 sharedPreferences = PreferenceManager
            .getDefaultSharedPreferences(getApplicationContext());
    // Check if message contains a data payload.
    if (remoteMessage.getData().size() > 0) {
        Intent myIntent = new Intent(MyFirebaseMessagingService.this, MyAlarmService.class);
        pendingIntent = PendingIntent.getService(MyFirebaseMessagingService.this, 0, myIntent, 0);
        AlarmManager alarmManager = (AlarmManager)getSystemService(ALARM_SERVICE);
        long currentTime = System.currentTimeMillis();

        int hour=sharedPreferences.getInt("Hour",9);
        int min=sharedPreferences.getInt("Min",0);
        Calendar calendar =  Calendar.getInstance();
        calendar.set(Calendar.HOUR_OF_DAY,hour);
        calendar.set(Calendar.MINUTE,min);



        alarmManager.set(AlarmManager.RTC_WAKEUP,calendar.getTimeInMillis() , pendingIntent);

这是 MyAlarmService onStartCommand:

 public int onStartCommand(Intent intent, int flags, int startId) {


        Uri uri = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);
        NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(this);
        mBuilder.setSmallIcon(R.mipmap.ic_launcher);
        mBuilder.setContentTitle("Figures");
        mBuilder.setContentText("New Figures has been updated!");
        mBuilder.setAutoCancel(true);
        mBuilder.setSound(uri);
        Intent resultIntent = new Intent(this, LoginActivity.class);
        TaskStackBuilder stackBuilder = TaskStackBuilder.create(this);
        stackBuilder.addParentStack(LoginActivity.class);
        stackBuilder.addNextIntent(resultIntent);
        PendingIntent resultPendingIntent = stackBuilder.getPendingIntent(0, PendingIntent.FLAG_UPDATE_CURRENT);
        mBuilder.setContentIntent(resultPendingIntent);
        NotificationManager mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
        mNotificationManager.notify(0, mBuilder.build());


    return super.onStartCommand(intent, flags, startId);
}

这里是Vb.net发送通知功能:

Private Function sendNotification() As Integer
    Try
        Dim url As String = "https://fcm.googleapis.com/fcm/send"
        Dim tRequest As WebRequest = WebRequest.Create(url)

        tRequest.Method = "post"
        tRequest.ContentType = "application/json"
        Dim data = New With { _
            Key .[to] =myTopic, _
                Key .TTL = "109", _
            Key .data = New With { _
                Key .body = "Updated", _
                Key .title = "Daily" _
            } _
        }
        Dim jssons As String = Newtonsoft.Json.JsonConvert.SerializeObject(data)
        Dim bytearray As Byte() = Encoding.UTF8.GetBytes(jssons)
        tRequest.Headers.Add(String.Format("Authorization: key={0}", MyKey))
        tRequest.Headers.Add(String.Format("Sender: id={0}", MySenderId))
        tRequest.ContentLength = bytearray.Length
        tRequest.ContentType = "application/json"
        Using datastream As Stream = tRequest.GetRequestStream()
            datastream.Write(bytearray, 0, bytearray.Length)

            Using tresponse As WebResponse = tRequest.GetResponse

                Using dataStreamResponse As Stream = tresponse.GetResponseStream()

                    Using tReader As StreamReader = New StreamReader(dataStreamResponse)

                        Dim sResponseFromServer As String = tReader.ReadToEnd()

                        Log(sResponseFromServer, w)
                    End Using
                End Using

            End Using
        End Using

    Catch ex As WebException
        Log(ex.Message, w)
        Return ex.Status



    End Try
    Log("Notification sent", w)
    Return 200

End Function 

有没有更好的发送通知的方法?AlarmManager 内部是否有问题? 谢谢。

您的代码似乎没问题。但是根据文档:

A common scenario for triggering an operation outside the lifetime of your app is syncing data with a server. This is a case where you might be tempted to use a repeating alarm. But if you own the server that is hosting your app's data, using Google Cloud Messaging (GCM) in conjunction with sync adapter is a better solution than AlarmManager. A sync adapter gives you all the same scheduling options as AlarmManager, but it offers you significantly more flexibility. For example, a sync could be based on a "new data" message from the server/device (see Running a Sync Adapter for details), the user's activity (or inactivity), the time of day, and so on. See the linked videos at the top of this page for a detailed discussion of when and how to use GCM and sync adapter.

来源:Scheduling Repeating Alarms documentation

或许您可以尝试使用同步适配器。还要检查您是否收到两次通知(我认为您没有)。或者检查闹钟是否设置为 30 分钟的重复周期。

希望对您有所帮助。

Return START_NOT_STICKY 来自 MyAlarmService.onStartCommand() 而不是调用 super 方法。超级方法returns START_STICKY,它会在服务被系统杀死后重新启动您的服务,重新触发通知。

另一个问题是为此使用服务总体上是否是个好主意。我会尝试 BroadcastReceiver,它不会在你完成后保留 运行。