设定alarm的话,大概需要用到如下几步:
1.创建一个Intent
2.创建一个PendingIntent
3.得到ALARM_SERVICE的AlarmManager
4.使用AlarmManager的set api,假如是取消,那就使用cancel
android自带的AP对于Alarm的管理是很巧妙的,例如Calendar,它基本上只会给系统设定一个alarm,等这个alarm结束之后,再设定下一个,这种模式很安全,也很有效,我认为这是一个值得学习的模式。
我之前一直有个疑问,设定的alarm,取消时,究竟要传怎样的PendingIntent系统才能知道我要取消的是哪一个
后来经过实验,我发现大概有这样几个参数会起决定作用,在我上面讲的四个步骤的第一步,Intent 的setData,setClass均能区别出,但是
putExtra却是没有用的。另外在PendingIntent.getBroadcast的第二个参数requestCode,查api,Google说这个参数没用的,但是其实它也是可以区别出不同的alarm。
上代码:
- package com.ianc.lily;
- import android.app.Activity;
- import android.app.AlarmManager;
- import android.app.PendingIntent;
- import android.content.Context;
- import android.content.Intent;
- import android.net.Uri;
- import android.os.Bundle;
- import android.util.Log;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.widget.Button;
- public class LaunchActivity extends Activity implements OnClickListener {
- static final String LILY_TEST_INTENT = "com.ianc.lilytestintent";
- static final String ID = "id";
- static final String TIME = "alarm_time";
- int id;
- Button addBtn;
- Button cancelBtn;
- /** Called when the activity is first created. */
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- id = 1;
- addBtn = (Button) findViewById(R.id.addbtn);
- cancelBtn = (Button) findViewById(R.id.cancelbtn);
- addBtn.setOnClickListener(this);
- cancelBtn.setOnClickListener(this);
- }
- @Override
- public void onClick(View v) {
- if (v == addBtn){
- AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
- Intent intent = new Intent(LILY_TEST_INTENT);
- intent.setData(Uri.parse("content://calendar/calendar_alerts/1"));
- intent.setClass(this, LilyReceiver.class);
- intent.putExtra(ID, id);
- long atTimeInMillis = System.currentTimeMillis() + 5000;
- intent.putExtra(TIME, atTimeInMillis);
- // intent.putExtra(LABEL, label);
- // intent.putExtra(TIME, atTimeInMillis);
- PendingIntent sender = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT);
- am.set(AlarmManager.RTC_WAKEUP, atTimeInMillis, sender);
- Log.i("lily","add alarm");
- }
- else if (v == cancelBtn){
- AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
- Intent intent = new Intent(LILY_TEST_INTENT);
- intent.setClass(this, LilyReceiver.class);
- intent.setData(Uri.parse("content://calendar/calendar_alerts/1"));
- // id = 2;
- // Log.i("lily","id = 2");
- // intent.putExtra(ID, id);
- PendingIntent sender = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_NO_CREATE);
- if (sender != null){
- Log.i("lily","cancel alarm");
- am.cancel(sender);
- }else{
- Log.i("lily","sender == null");
- }
- }
- }
- }
1.Intent中的setData使用的值,必须和cancel中intent里面setData用的值一样,否则点击cancel是没法取消掉的,或者你不setData,那也是没法取消的
2.Intent中的setClass使用的值也必须和cancel众intent使用的一样,不然也cancel不掉,不setClass也cancel不了
3.Intent中的putExtra是无效的,根本不起区别的作用
4.getBroadcast的第二个参数,一般的ap都是写0,其实假如你前面的intent只有setAction过,那么单纯用reqeustCode也是可以区别不同的alarm的。
5.通常cancel之前可以先用PendingIntent.FLAG_NO_CREATE来判断之前是不是设定了这个alarm,假如没设定,那就不要去调用cancel
补充工程中的次要代码:
AndroidManifest.xml
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.ianc.lily"
- android:versionCode="1"
- android:versionName="1.0">
- <application android:icon="@drawable/icon" android:label="@string/app_name">
- <activity android:name=".LaunchActivity"
- android:label="@string/app_name">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <receiver android:name="LilyReceiver">
- <intent-filter>
- <action android:name="com.ianc.lilytestintent" />
- <data android:mimeType="vnd.android.cursor.item/calendar-alert" />
- </intent-filter>
- </receiver>
- </application>
- </manifest>
LilyReceiver.java
- package com.ianc.lily;
- import android.content.BroadcastReceiver;
- import android.content.Context;
- import android.content.Intent;
- import android.util.Log;
- import android.widget.Toast;
- public class LilyReceiver extends BroadcastReceiver {
- @Override
- public void onReceive(Context context, Intent intent) {
- String action = intent.getAction();
- int id = intent.getIntExtra("id", -1);
- long alarmtime= intent.getLongExtra("alarm_time", -1);
- Log.i("lily","received action = "+action+", id = "+id+ ", alarmtime = "+alarmtime);
- Toast.makeText(context, "received action = "+action+", id = "+id, Toast.LENGTH_SHORT).show();
- }
- }