这几天因为案子需要,研究了一下alarm的设定和取消。
众所周知,设定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
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(); } }