在Android中创建定时通知(例如,针对事件)

时间:2012-01-06 01:03:03

标签: android notifications alarmmanager

对于某些Android应用程序,我想集成以下功能: 用户可以定义他想要被提醒的时间。当时间到了,应用程序应该在通知栏中创建通知,即使此时用户没有使用该应用程序。

为此,需要查看AlarmManager,NotificationManager和Notification.Builder类,对吧?

那么如何提前创建定时通知?我的代码(到目前为止)是这样的:

将其添加到AndroidManifest以注册广播接收器:

<receiver android:name="AlarmNotificationReceiver"></receiver>

创建一个新的类文件来处理它收到的警报:

public class AlarmNotificationReceiver extends BroadcastReceiver {

    public void onReceive(Context context, Intent intent) {
        Bundle extras = intent.getExtras();
        if (extras != null) {
            String additionalData = extras.getString("displayText");
            // show the notification now
            NotificationManager mNotificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
            Notification mNotification = new Notification(R.drawable.ic_launcher, context.getString(R.string.app_name), System.currentTimeMillis());
            PendingIntent pi = PendingIntent.getActivity(context, 0, new Intent(context, MainActivity.class), 0); // open MainActivity if the user selects this notification
            mNotification.setLatestEventInfo(context, context.getString(R.string.app_name), additionalData, pi);
            mNotification.flags |= Notification.FLAG_AUTO_CANCEL | Notification.DEFAULT_SOUND;
            mNotificationManager.notify(1, mNotification);
        }
    }

}

使用此代码(例如在MainActivity中)将警报设置为3秒后:

    Intent i = new Intent(this, AlarmNotificationReceiver.class);
    i.putExtra("displayText", "sample text");
    PendingIntent pi = PendingIntent.getBroadcast(this.getApplicationContext(), 234324246, i, 0);
    AlarmManager mAlarm = (AlarmManager) getApplicationContext().getSystemService(Context.ALARM_SERVICE);
    mAlarm.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+3*1000, pi);

我需要做些什么来改变这项工作?谢谢!

这两个问题是:

  1. 当我在代码中更改通知时,通知的文本不会更改。它只会在我更改PendingIntent.getBroadcast(...)中的requestCode时更改。这个请求代码到底是什么?可以是随机值还是0?
  2. 重新启动手机后,“计划”通知或警报消失。但是现在我已经看到这是正常行为,对吗?我怎么能绕过这个?

2 个答案:

答案 0 :(得分:4)

对第1部分不确定,但对于第2部分,一般方法是拦截BOOT_COMPLETED意图并使用它重新注册所有警报。遗憾的是,对于您在警报管理器中注册的每个警报,您都必须将其存储在应用程序的数据库中。

因此,您需要一个广播接收器来拦截BOOT_COMPLETED意图:

public class BootReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // get your stored alarms from your database
        // reregister them with the alarm manager
    }
}

要获取BOOT_COMPLETED意图,您必须在清单中添加以下权限:

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

还需要使用以下intent过滤器在您的清单中注册BootReceiver:

    <receiver android:enabled="true" android:name=".receiver.BootReceiver"
    android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
        <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
                <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </receiver>

重要的是要注意,如果您的应用程序安装到SD卡,它永远不会收到BOOT_COMPLETED意图。此外,值得注意的是,这种实现有点天真,因为它在启动时立即执行代码,这会在启动时降低用户的手机速度。因此,我建议在拦截启动意图后将执行延迟几分钟。

答案 1 :(得分:1)

我个人会在没有广播接收器的情况下这样做。我会让AlarmManager触发启动单独Activity的意图,而不是接收器。然后这个新的活动可以为您发出通知。我不确定这是否是一种更好的方式,但对我来说似乎并不那么复杂。

编辑:服务可能比活动

更好

在您的MainActivity中:

Intent i = new Intent(getBaseContext(), NotificationService.class);
PendingIntent pi = PendingIntent.getService(getBaseContext(), 0, i, 0);
AlarmManager mAlarm = (AlarmManager) Context.getSystemService(Context.ALARM_SERVICE);
mAlarm.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+10*60*1000, pi);

您的服务:

public class NotificationService extends Service {

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

//Create the notification here

return START_NOT_STICKY;
}

你的宣言:

<service android:name="com.android.yourpath.NotificationService"></service>