startForegroundService() 没有调用 startForeground(),但确实调用了

Posted

技术标签:

【中文标题】startForegroundService() 没有调用 startForeground(),但确实调用了【英文标题】:startForegroundService() did not call startForeground(), but it did 【发布时间】:2018-10-25 02:30:23 【问题描述】:

我的 android 服务中有 Context.startForegroundService() did not then call Service.startForeground(),但我不知道为什么会这样。

我的应用程序用于媒体流,只有当您从前台通知暂停(将其变为常规通知),然后将通知滑开时才会出现此错误,这旨在停止我的服务。

这是调用startForegroundServicestartForegroundstopForeground 方法的唯一方法:

private void configureServiceState(long action) 
        if (action == PlaybackStateCompat.ACTION_PLAY) 
            if (!mServiceInStartedState) 
                ContextCompat.startForegroundService(
                        StreamingService.this,
                        new Intent(
                                StreamingService.this,
                                StreamingService.class));
                mServiceInStartedState = true;
             startForeground(NOTIFICATION_ID,
                    buildNotification(PlaybackStateCompat.ACTION_PAUSE));
         else if (action == PlaybackStateCompat.ACTION_PAUSE) 
            stopForeground(false);

            NotificationManager mNotificationManager
                    = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);

            assert mNotificationManager != null;
            mNotificationManager
                    .notify(NOTIFICATION_ID,
                            buildNotification(PlaybackStateCompat.ACTION_PLAY));
         else if (action == PlaybackStateCompat.ACTION_STOP) 
            mServiceInStartedState = false;
            stopForeground(true);
            stopSelf();
        
    

这里是设置我的通知的删除意图的地方:

.setDeleteIntent(
                        MediaButtonReceiver.buildMediaButtonPendingIntent(
                                this, PlaybackStateCompat.ACTION_STOP));

这个configureServiceState(long action) 方法只能从我的MediaSession 回调中调用:onPlayonPauseonStop... 显然该操作是要执行的预期操作。

从我的 UI 执行 onStop 或从 UI 调用 onPause 后跟 onStop(镜像清除通知所需的操作)时,仅从通知中不会发生错误。

我能找到的关于此错误的所有信息是,它可能发生在您调用 startForegroundService 但未在 5 秒内调用 startForeground 时......但是在唯一一次 startForegroundService 之后立即调用 startForeground被调用。

此外,onPlaybackStateChange 将在onStop 方法中进入我的“正在播放”活动,这会触发该活动运行finish(),这样服务就不会以这种方式重新启动。

我在这里错过了什么?

其他细节:

我的“正在播放”活动未部分重新启动该服务,因为代码执行从未到达其任何方法。

在产生错误之前,代码执行似乎也永远不会重新输入configureServiceState

如果我在最后一个可能的点(我的服务的onStartCommand 中的MediaButtonReceiver.handleIntent(mMediaSession, intent);)添加断点,在此处暂停执行并尝试调试会导致调试器在暂停后不久断开连接

李>

为前台和常规通知尝试不同的通知渠道也没有什么不同

将暂停的通知从锁定屏幕上滑开不会导致错误,只有在手机解锁时才会出现;无论我的应用是否真正打开

完全例外:

android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground()
                      at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1870)
                      at android.os.Handler.dispatchMessage(Handler.java:105)
                      at android.os.Looper.loop(Looper.java:164)
                      at android.app.ActivityThread.main(ActivityThread.java:6809)
                      at java.lang.reflect.Method.invoke(Native Method)
                      at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
                      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)

我的测试设备运行的是 Android 8.0,项目的最小 API 是 21,目标 API 是 27。设备 API 是 26。

【问题讨论】:

我最近在创建媒体服务时遇到了这个确切的问题(尽管这可能并不完全相关)。请将您的 Android 版本和 API 级别添加到您的问题中,我会看看是否可以提供帮助。 谢谢。我已经添加了这一点以及我在调试过程中发现的另一件事。 @transiti0nary if (action == PlaybackStateCompat.ACTION_PLAY) 如果条件在 5 秒内执行,这是吗?我的意思是这个条件true 和 ACTION_PLAY 在 5 秒内触发? 服务启动后 5 秒内?我不知道。为了安全起见,我也可以在onCreate 内触发它,但是是否有检查前台通知是否存在的功能,所以当播放从暂停移动到播放时,我仍然可以在此功能内启动它(当onCreate 获胜时不叫? 【参考方案1】:

您需要为 Android O API 26 及更高版本使用 NotificationChannel,否则您将收到您遇到的错误。请参阅 Android 文档中的以下声明 - Create and Manage Notification Channels:

从 Android 8.0(API 级别 26)开始,所有通知都必须分配给一个频道。

这是我用于构建媒体通知的方法的摘录(从中获取您需要的内容)。您可以看到有一个针对 Android O 设备的检查,其中包含处理这种情况的特定方法:

private fun compileNotification(context: Context, action: NotificationCompat.Action, mediaSession: MediaSessionCompat, controller: MediaControllerCompat, mMetadata: MediaMetadataCompat, art: Bitmap?, mPlaybackState: PlaybackStateCompat) 

    val description = mMetadata.description

    // https://***.com/questions/45395669/notifications-fail-to-display-in-android-oreo-api-26
    @TargetApi(26)
    if(Utils.hasO()) 
        val channelA = mNotificationManager?.getNotificationChannel(NotificationChannelID.MEDIA_SERVICE.name)

        if(channelA == null) 
            val channelB = NotificationChannel(NotificationChannelID.MEDIA_SERVICE.name,
                    "MediaService",
                    NotificationManager.IMPORTANCE_DEFAULT)
            channelB.setSound(null, null)

            mNotificationManager?.createNotificationChannel(channelB)
        
    

    val notificationBuilder = if(Utils.hasLollipop()) 
        NotificationCompat.Builder(context, NotificationChannelID.MEDIA_SERVICE.name)
     else 
        NotificationCompat.Builder(context)
    

    notificationBuilder
            .setStyle(android.support.v4.media.app.NotificationCompat.MediaStyle()
                    // Show actions 0,2,4 in compact view
                    .setShowActionsInCompactView(0,2,4)
                    .setMediaSession(mediaSession.sessionToken))
            .setSmallIcon(R.drawable.logo_icon)
            .setShowWhen(false)
            .setContentIntent(controller.sessionActivity)
            .setContentTitle(description.title)
            .setContentText(description.description)
            .setLargeIcon(art)
            .setVisibility(NotificationCompat.VISIBILITY_PUBLIC)
            .setOngoing(mPlaybackState.state == PlaybackStateCompat.STATE_PLAYING)
            .setOnlyAlertOnce(true)

            if(!Utils.hasLollipop()) 
                notificationBuilder
                        .setStyle(android.support.v4.media.app.NotificationCompat.MediaStyle()
                                // Show actions 0,2,4 in compact view
                                .setShowActionsInCompactView(0,2,4)
                                .setMediaSession(mediaSession.sessionToken)
                                .setShowCancelButton(true)
                                .setCancelButtonIntent(MediaButtonReceiver.buildMediaButtonPendingIntent(context,
                                        PlaybackStateCompat.ACTION_STOP)))
                        // Stop the service when the notification is swiped away
                        .setDeleteIntent(MediaButtonReceiver.buildMediaButtonPendingIntent(context,
                                PlaybackStateCompat.ACTION_STOP))
            

    notificationBuilder.addAction(NotificationCompat.Action(
            R.drawable.exo_controls_previous,
            "Previous",
            MediaButtonReceiver.buildMediaButtonPendingIntent(context,
                    PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS)))
    notificationBuilder.addAction(NotificationCompat.Action(
            R.drawable.ic_replay_10_white_24dp,
            "Rewind",
            MediaButtonReceiver.buildMediaButtonPendingIntent(context,
                    PlaybackStateCompat.ACTION_REWIND)))

    notificationBuilder.addAction(action)

    notificationBuilder.addAction(NotificationCompat.Action(
            R.drawable.ic_forward_10_white_24dp,
            "Fast Foward",
            MediaButtonReceiver.buildMediaButtonPendingIntent(context,
                    PlaybackStateCompat.ACTION_FAST_FORWARD)))
    notificationBuilder.addAction(NotificationCompat.Action(
            R.drawable.exo_controls_next,
            "Next",
            MediaButtonReceiver.buildMediaButtonPendingIntent(context,
                    PlaybackStateCompat.ACTION_SKIP_TO_NEXT)))

    (context as MediaService).startForeground(NOTIFICATION_ID, notificationBuilder.build())

在您的onStop() 回调中,您需要在服务中调用stopSelf()。然后,当您的服务 onDestroy() 方法被调用时,您需要清理一些东西(适用于您的情况),如下所示:

override fun onDestroy() 
    super.onDestroy()

    abandonAudioFocus()
    unregisterReceiver(mNoisyReceiver)

    //Deactivate session
    mSession.isActive = false
    mSession.release()

    NotificationManagerCompat.from(this).cancelAll()

    if(mWiFiLock?.isHeld == true) mWiFiLock?.release()

    stopForeground(true)

我没有包含上述某些方法的详细信息,但方法名称应该是自我注释的 - 如果我可以包含更多详细信息,请告诉我。其中一些可能是矫枉过正,但在您的情况下可能会解决问题。

我很确定这会解决您的问题。如果没有,我还有一些想法。

【讨论】:

谢谢,如果可以的话,我会在今天晚些时候试一试。不过有一件事,我的通知构建功能确实运行此处显示的通知通道功能:developer.android.com/training/notify-user/channels 是在做同样的事情吗?我注意到的另一件事是,如果!Utils.hasLollipop(),我的构建器没有额外的setStyle,并且始终使用此构造器NotificationCompat.Builder(context, NOTIFICATION_CHANNEL),而不管hasLollipop。会不会跟那个有关? 在这种情况下,它可能是一个不同的问题 - 我仍然认为如果不是太麻烦的话,可能值得尝试上述方法。通常兼容库实际上会处理不同的 Android 版本检查。 @transiti0nary 我添加了一些关于如何在媒体服务中处理onStop() 方法的详细信息。如果这有帮助,请告诉我。 stopSelf(); 之前调用stopForeground(true); 的代码中可能存在问题 - stopSelf() 不会立即终止服务。 如果这个或任何答案解决了您的问题或以任何方式帮助您,请考虑通过单击复选标记接受它。这向更广泛的社区表明您已经找到了解决方案,并为回答者和您自己提供了一些声誉。没有义务这样做。【参考方案2】:

只是在这上面浪费了太多时间。我不确定这就是您所遇到的,但就我而言,我一直收到此异常,因为我的 NOTIFICATION_ID0 ...使用任何其他值似乎可以解决此问题。-_-

【讨论】:

虽然这里不是这样,但我也经历过!花了很长时间才弄明白。【参考方案3】:

我也遇到了同样的问题。问题是在通知项目上使用PendingIntent.getForegroundService() 来更新媒体服务。仅使用 PendingIntent.getService 为我解决了这个问题。

【讨论】:

有趣,但我不认为我直接使用该方法。通知交互发送 MediaButton 事件,这些事件由我的服务接收并处理 我在 Android 8、9 上遇到同样的问题,用 PendingIntent.getService 替换 PendingIntent.getForegroundService() 解决它

以上是关于startForegroundService() 没有调用 startForeground(),但确实调用了的主要内容,如果未能解决你的问题,请参考以下文章

startForegroundService - 无法使用 setContentIntent + addAction?

startForegroundService() 未在 2 台三星设备上启动服务

想要隐藏通知但获取 Context.startForegroundService() 并没有调用 Service.startForeground()

将 Context.startForegroundService(Intent) 而不是 Context.startService(Intent) 用于前台服务有啥好处吗?

绑定服务崩溃与“Context.startForegroundService() 没有然后调用 Service.startForeground()”错误

Context.startForegroundService 然后没有调用 Service.startForeground