使用PendingIntent结合 alarmManager 发送定时广播,收到广播的时候显示notification的功能,遇到了好多坑。
Intent intent = new Intent(KeyVaule.CLOCK_IN_ACTION_NAME);
intent.putExtra(KeyName.SOURCE, KeyVaule.MANUAL_CLOCK_IN);
intent.putExtra(KeyName.CARD_NO, cardNo);
intent.putExtra(KeyName.TARGET_TIME, targetTime + "");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
ComponentName componentName = new ComponentName(context.getPackageName(),context.getPackageName() + ".receiver.AlarmReceiver");
intent.setComponent(componentName);
}
//todo 问题:flag为PendingIntent.FLAG_UPDATE_CURRENT 可以更新发送intent的内容
//todo 问题:requestcode 不变的话,最新的intent会覆盖之前的
PendingIntent pi = PendingIntent.getBroadcast(context, 3, intent, PendingIntent.FLAG_UPDATE_CURRENT);
//PendingIntent 和 闹钟 结合发送 定时广播
AlarmManager am = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
am.setExact(AlarmManager.RTC_WAKEUP, targetTime, pi);
} else {
am.set(AlarmManager.RTC_WAKEUP, targetTime, pi);
}
public class AlarmReceiver extends BroadcastReceiver {
private static final long TEN_HOUR = 10 * 60 * 60 * 1000;
@Override
public void onReceive(Context context, final Intent intent) {
showNotification(context);
}
private void showNotification(Context context) {
String time = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.CHINA).format(System.currentTimeMillis());
NotificationCompat.Builder builder;
NotificationManager notifManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
String id = "channel_01";
String name = "channel_name";
if (notifManager == null) {
notifManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
int importance = NotificationManager.IMPORTANCE_LOW;
NotificationChannel mChannel = notifManager.getNotificationChannel(id);
if (mChannel == null) {
mChannel = new NotificationChannel(id, name, importance);
mChannel.setDescription("");
notifManager.createNotificationChannel(mChannel);
}
builder = new NotificationCompat.Builder(context, id);
builder.setContentTitle("收到打卡闹钟") // required
.setSmallIcon(R.mipmap.clock) // required
.setContentText("打卡时间:" + time) // required
.setDefaults(Notification.DEFAULT_ALL)
.setAutoCancel(true)
.setTicker("收到打卡闹钟");
} else {
builder = new NotificationCompat.Builder(context);
builder.setContentTitle("收到打卡闹钟") // required
.setSmallIcon(R.mipmap.clock) // required
.setContentText("打卡时间:" + time) // required
.setDefaults(Notification.DEFAULT_ALL)
.setAutoCancel(true)
.setTicker("收到打卡闹钟")
.setPriority(Notification.PRIORITY_HIGH);
}
Notification notification = builder.build();
notifManager.notify(111, notification);
}
}
8.0以上隐式广播限制
以前只要设置action隐式广播就可以接收到,8.0以上不可以了,需要指明接收器的包名。
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
ComponentName componentName = new ComponentName(context.getPackageName(),context.getPackageName() + ".receiver.AlarmReceiver");
intent.setComponent(componentName);
}
pendingintent 的requestcode和flag影响
pendingIntent结合闹钟 使用遇到的最大的问题是,如果把APPkill了,receiver就接收不到广播了!!!应该说闹钟到了,广播发送了,但是receiver被销毁了啊,接收不到广播了??求路过的大神给解释下!
// 获取 Broadcast 关联的 PendingIntent
PendingIntent.getBroadcast(Context context, int requestCode, Intent intent, int flags)
// 获取 Activity 关联的 PendingIntent
PendingIntent.getActivity(Context context, int requestCode, Intent intent, int flags)
PendingIntent.getActivity(Context context, int requestCode, Intent intent, int flags, Bundle options)
// 获取 Service 关联的 PendingIntent
PendingIntent.getService(Context context, int requestCode, Intent intent, int flags)
1、使用alarmManager,通过PendingIntent向broadcastReceiver传递内容。
第二个参数requestCode相同的话后面的定时器会将前面的定时器"覆盖"掉,只会启动最后一个定时器,
所以同一时间的定时器可以用同一个requestCode,不同时间的定时器用不同的requestCode。
2、使用notification,通过PendingIntent向Activity传递内容。
第二个参数requestCode不同,以及设置最后一个flags参数为PendingIntent.FLAG_UPDATE_CURRENT可以
保证每次点击notification进入activity,传的参数都是当前这个notification传过去的(Intent里附带
的参数)。如果requestCode相同,最后一个参数为PendingIntent.FLAG_UPDATE_CURRENT,那么所有未
点击的notification里的参数都会被更新为最后一个传过去的,也就是最新的。
flags的取值有四个:
FLAG_ONE_SHOT:获取的PendingIntent只能使用一次
FLAG_NO_CREATE:利用FLAG_NO_CREAT获取的PendingIntent,若描述的Intent不存在则返回NULL值
FLAG_CANCEL_CURRENT:如果描述的PendingIntent已经存在,则在产生新的Intent之前会先取消掉当前的
FLAG_UPDATE_CURRENT:能够新new一个 Intent
private void showNotification(Context context) {
String time = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.CHINA).format(System.currentTimeMillis());
NotificationCompat.Builder builder;
NotificationManager notifManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
String id = "channel_01";
String name = "channel_name";
if (notifManager == null) {
notifManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
int importance = NotificationManager.IMPORTANCE_LOW;
NotificationChannel mChannel = notifManager.getNotificationChannel(id);
if (mChannel == null) {
mChannel = new NotificationChannel(id, name, importance);
mChannel.setDescription("");
notifManager.createNotificationChannel(mChannel);
}
builder = new NotificationCompat.Builder(context, id);
builder.setContentTitle("收到闹钟") // required
.setSmallIcon(R.mipmap.clock) // required
.setContentText("时间:" + time) // required
.setDefaults(Notification.DEFAULT_ALL)
.setAutoCancel(true)
.setTicker("收到闹钟");
} else {
builder = new NotificationCompat.Builder(context);
builder.setContentTitle("收到闹钟") // required
.setSmallIcon(R.mipmap.clock) // required
.setContentText("时间:" + time) // required
.setDefaults(Notification.DEFAULT_ALL)
.setAutoCancel(true)
.setTicker("收到闹钟")
.setPriority(Notification.PRIORITY_HIGH);
}
Notification notification = builder.build();
notifManager.notify(111, notification);
}