设定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();
}
}