Sim*_*mba 5 permissions android alarmmanager android-pendingintent
美好的一天,我在 Android 应用程序中内置了一个计时器,可以在特定时间后发送消息。为此,我添加了一个带有待处理意图的 AlarmManager。当我打开应用程序时,一切都正常。它也适用于锁定屏幕。但是,当我关闭应用程序时,挂起的意图将不再激活,并且在给定时间不会出现任何消息。这是我的警报管理器:
Intent notifyIntent = new Intent(getContext(), MyReceiver.class);
alarmManager = (AlarmManager) getContext().getSystemService(Context.ALARM_SERVICE);
pendingIntent = PendingIntent.getBroadcast
(getContext(), NOTIFICATION_ID, notifyIntent, PendingIntent.FLAG_UPDATE_CURRENT);
//when the alarm gets activated:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
alarmManager.setExact(AlarmManager.RTC_WAKEUP,System.currentTimeMillis()+delta_time2, pendingIntent);
}
else {
alarmManager.set(AlarmManager.RTC_WAKEUP,System.currentTimeMillis()+delta_time2, pendingIntent);
}
Run Code Online (Sandbox Code Playgroud)
在 IntentService 类中(声明和调用通知的地方):
Intent notifyIntent = new Intent(this, BigTextMainActivity.class);
notifyIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
PendingIntent notifyPendingIntent =
PendingIntent.getActivity(
this,
0,
notifyIntent,
PendingIntent.FLAG_UPDATE_CURRENT
);
Intent snoozeIntent = new Intent(this, BigTextIntentService.class);
snoozeIntent.setAction(BigTextIntentService.ACTION_SNOOZE);
PendingIntent snoozePendingIntent = PendingIntent.getService(this, 0, snoozeIntent, 0);
NotificationCompat.Action snoozeAction =
new NotificationCompat.Action.Builder(
R.drawable.ic_alarm_white_48dp,
"Snooze",
snoozePendingIntent)
.build();
Intent dismissIntent = new Intent(this, BigTextIntentService.class);
dismissIntent.setAction(BigTextIntentService.ACTION_DISMISS);
PendingIntent dismissPendingIntent = PendingIntent.getService(this, 0, dismissIntent, 0);
NotificationCompat.Action dismissAction =
new NotificationCompat.Action.Builder(
R.drawable.ic_cancel_white_48dp,
"Dismiss",
dismissPendingIntent)
.build();
NotificationCompat.Builder notificationCompatBuilder =
new NotificationCompat.Builder(
getApplicationContext(), notificationChannelId);
GlobalNotificationBuilder.setNotificationCompatBuilderInstance(notificationCompatBuilder);
Notification notification = notificationCompatBuilder
.setStyle(bigTextStyle)
.setContentTitle(bigTextStyleReminderAppData.getContentTitle())
.setContentText(bigTextStyleReminderAppData.getContentText())
.setSmallIcon(R.drawable.ic_launcher)
.setLargeIcon(BitmapFactory.decodeResource(
getResources(),
R.drawable.ic_alarm_white_48dp))
.setContentIntent(notifyPendingIntent)
.setDefaults(NotificationCompat.DEFAULT_ALL)
.setColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimary))
.setCategory(Notification.CATEGORY_REMINDER)
.setPriority(bigTextStyleReminderAppData.getPriority())
.setVisibility(bigTextStyleReminderAppData.getChannelLockscreenVisibility())
.addAction(snoozeAction)
.addAction(dismissAction)
.build();
//I am not quite shure, if I declared the foregroundservice for the notification at the right place
startForeground(NOTIFICATION_ID, notification);
mNotificationManagerCompat.notify(NOTIFICATION_ID, notification);
Vibrator vibrator = (Vibrator) getApplication().getSystemService(Context.VIBRATOR_SERVICE);
vibrator.vibrate(2 * 1000);
Run Code Online (Sandbox Code Playgroud)
这些是我的权限:
<uses-permission android:name="android.permission.WAKE_LOCK"/>
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<uses-permission android:name="android.permission.VIBRATE"/>
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
Run Code Online (Sandbox Code Playgroud)
感谢您的帮助!(顺便说一句,如果这个问题很愚蠢或其他什么,那是因为我没有真正处理这个主题,只是想快速为我的应用程序启动警报,提前抱歉)
根据您的评论,看起来警报正在按预期触发,但BroadcastReceiver在尝试启动Service. 最可能的原因是您的设备对后台处理有额外的限制。一些设备(主要是低端设备和中国制造的设备)对允许哪些应用程序“在后台”运行有限制。如果您的应用程序不在允许的应用程序“白名单”中,Android 将不允许在后台启动该应用程序的组件。
请参阅https://developer.android.com/about/versions/oreo/background了解背景Service限制,并参阅/sf/answers/3183767611/作为答案的示例,其中我讨论了以下问题某些制造商限制后台处理。
注意:由于您无法在 Android 8 及更高版本上启动后台Service(由于限制),因此您可以将其启动Service为前台Service,如下所示:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
context.startForegroundService(intent)
} else {
context.startService(intent)
}
Run Code Online (Sandbox Code Playgroud)