本节引言:
本节主要介绍的是Android系统服务中的---AlarmManager(闹钟服务),
除了开发手机闹钟外,更多的时候是作为一个全局的定时器,通常与Service
结合,在特定时间启动其他的组件!本节就来对这个AlarmManager来进行解析
同时通过小闹钟与自动换壁纸来演示这个AlarmManager的用法,好了,开始本节的
内容吧!
本节正文:
1.概念与相关属性方法:
AlarmManager:
1、AlarmManager,顾名思义,就是“提醒”,是Android中常用的一种系统级别的提示服务,在特定的时刻为我们广播一个指定的Intent。简单的说就是我们设定一个时间,然后在该时间到来时,AlarmManager为我们广播一个我们设定的Intent,通常我们使用 PendingIntent,PendingIntent可以理解为Intent的封装包,简单的说就是在Intent上在加个指定的动作。在使用Intent的时候,我们还需要在执行startActivity、startService或sendBroadcast才能使Intent有用。而PendingIntent的话就是将这个动作包含在内了。
定义一个PendingIntent对象。
PendingIntent pi = PendingIntent.getBroadcast(this,0,intent,0);
2、AlarmManager的常用方法有三个:
(1)set(int type,long startTime,PendingIntent pi);
该方法用于设置一次性闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟执行时间,第三个参数表示闹钟响应动作。
(2)setRepeating(int type,long startTime,long intervalTime,PendingIntent pi);
该方法用于设置重复闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟首次执行时间,第三个参数表示闹钟两次执行的间隔时间,第三个参数表示闹钟响应动作。
(3)setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi);
该方法也用于设置重复闹钟,与第二个方法相似,不过其两个闹钟执行的间隔时间不是固定的而已。
3、三个方法各个参数详悉:
(1)int type: 闹钟的类型,常用的有5个值:AlarmManager.ELAPSED_REALTIME、 AlarmManager.ELAPSED_REALTIME_WAKEUP、AlarmManager.RTC、 AlarmManager.RTC_WAKEUP、AlarmManager.POWER_OFF_WAKEUP。
AlarmManager.ELAPSED_REALTIME表示闹钟在手机睡眠状态下不可用,该状态下闹钟使用相对时间(相对于系统启动开始),状态值为3;
AlarmManager.ELAPSED_REALTIME_WAKEUP表示闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟也使用相对时间,状态值为2;
AlarmManager.RTC表示闹钟在睡眠状态下不可用,该状态下闹钟使用绝对时间,即当前系统时间,状态值为1;
AlarmManager.RTC_WAKEUP表示闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟使用绝对时间,状态值为0;
AlarmManager.POWER_OFF_WAKEUP表示闹钟在手机关机状态下也能正常进行提示功能,所以是5个状态中用的最多的状态之一,该状态下闹钟也是用绝对时间,状态值为4;不过本状态好像受SDK版本影响,某些版本并不支持;
(2)long startTime: 闹钟的第一次执行时间,以毫秒为单位,可以自定义时间,不过一般使用当前时间。需要注意的是,本属性与第一个属性(type)密切相关,如果第一个参数对 应的闹钟使用的是相对时间(ELAPSED_REALTIME和ELAPSED_REALTIME_WAKEUP),那么本属性就得使用相对时间(相对于 系统启动时间来说),比如当前时间就表示为:SystemClock.elapsedRealtime();如果第一个参数对应的闹钟使用的是绝对时间 (RTC、RTC_WAKEUP、POWER_OFF_WAKEUP),那么本属性就得使用绝对时间,比如当前时间就表示 为:System.currentTimeMillis()。
(3)long intervalTime:对于后两个方法来说,存在本属性,表示两次闹钟执行的间隔时间,也是以毫秒为单位。
(4)PendingIntent pi: 绑定了闹钟的执行动作,比如发送一个广播、给出提示等等。PendingIntent是Intent的封装类。需要注意的是,如果是通过启动服务来实现闹钟提 示的话,PendingIntent对象的获取就应该采用Pending.getService(Context c,int i,Intent intent,int j)方法;如果是通过广播来实现闹钟提示的话,PendingIntent对象的获取就应该采用 PendingIntent.getBroadcast(Context c,int i,Intent intent,int j)方法;如果是采用Activity的方式来实现闹钟提示的话,PendingIntent对象的获取就应该采用 PendingIntent.getActivity(Context c,int i,Intent intent,int j)方法。如果这三种方法错用了的话,虽然不会报错,但是看不到闹钟提示效果。
第二个参数i 一定要是唯一的,比如不同的ID之类的,(如果系统需要多个定时器的话)。
关于AlarmManager具体用法见:http://blog.csdn.net/wangxingwu_314/article/details/8060312
Calendar.getInstance():
获取系统当前的日期年,月,日和getInstance()
package testdate; import java.util.*; import java.util.Calendar; public class TestMonth_2{ public static void main(String[]args){ Calendar cal = Calendar.getInstance(); int year = cal.get(Calendar.YEAR); System.out.println(year+"年"); int month = cal.get(Calendar.MONTH) + 1; System.out.println(month+"月"); int date = cal.get(Calendar.DATE); System.out.println(date+"日"); } }
输出: 2009年 10月 19日
Calendar 是抽象类
Calendar提供了一个类方法 getInstance,以获得此类型的一个通用的对象。Calendar的 getInstance方法回一个 Calendar对象,其日历字段已由当前日期和时间初始化:
Calendar rightNow = Calendar.getInstance();
关于getInstance()可以看看:
http://topic.csdn.net/u/20070301/11/b3f08634-e329-436c-9e27-ad3edc876028.html
pendingIntent:
1 来自百度知道:
PendingIntent的getActivity方法,第一个参数是上下文,没啥好说的,第二个参数 requestCode,这个后面说,第三个参数是 Intent,用来存储信息,第四个参数是对参数的操作标识,常用的就是FLAG_CANCEL_CURRENT和FLAG_UPDATE_CURRENT。
当使用FLAG_UPDATE_CURRENT时:
PendingIntent.getActivity(context, 0, notificationIntent,PendingIntent.FLAG_CANCEL_CURRENT时);
FLAG_UPDATE_CURRENT会更新之前PendingIntent的消息,比如,你推送了消息1,并在其中的Intent中putExtra了一个值“ABC”,在未点击该消息前,继续推送第二条消息,并在其中的Intent中putExtra了一个值“CBA”,好了,这时候,如果你单击消息1或者消息2,你会发现,他俩个的Intent中读取过来的信息都是“CBA”,就是说,第二个替换了第一个的内容
当使用FLAG_CANCEL_CURRENT时:
依然是上面的操作步骤,这时候会发现,点击消息1时,没反应,第二条可以点击。
导致上面两个问题的原因就在于第二个参数requestCode,当requestCode值一样时,后面的就会对之前的消息起作用,所以为了避免影响之前的消息,requestCode每次要设置不同的内容。
2 来自http://blog.163.com/hnaylgqde@126/blog/static/11031970220113955852106/
pendingIntent字面意义:等待的,未决定的Intent。
要得到一个pendingIntent对象,使用方法类的静态方法getActivity(Context, int, Intent, int)
,getBroadcast(Context, int, Intent, int)
,getService(Context, int, Intent, int) 分别对应着Intent的3个行为,跳转到一个activity组件、打开一个广播组件和打开一个服务组件。 参数有4个,比较重要的事第三个和第一个,其次是第四个和第二个。可以看到,要得到这个对象,必须传入一个Intent作为参数,必须有context作为参数。
参数有4个,比较重要的事第三个和第一个,其次是第四个和第二个。可以看到,要得到这个对象,必须传入一个Intent作为参数,必须有context作为参数。
pendingIntent是一种特殊的Intent。主要的区别在于Intent的执行立刻的,而pendingIntent的执行不是立刻的。 pendingIntent执行的操作实质上是参数传进来的Intent的操作,但是使用 pendingIntent的目的在于它所包含的Intent的操作的执行是需要满足某些条件的。
主要的使用的地方和例子:通知Notificatio的发送,短消息SmsManager的发送和警报器AlarmManager的执行等等。
TimePickerDialog:pendingIntent
详细参数:
TimePickerDialog(Context context,TimePickerDialog.OnTimeSetListenercallBack,inthourOfDay,
int minute, boolean is24HourView)
参数说明:
Context:运行组件的Activity。
callBack:用户选择好时间后,通知应用的回调函数。
hourOfDay:初始的小时。
Minute:初始的分钟。
is24HourView:是否使用24小时制,返回为boolean,true为24小时制,false为12小时。
2.例子演示
好了,光看不练可不行,下面就来写两个简单的例子:
分别是定时闹钟提醒的实现与定时更换手机壁纸,一个是调用Activity,一个是调用Service
①一个简单的闹钟:
MainActivity.java:
packagecom.jay.example.alarmmanagerdemo;
importjava.util.Calendar;
importandroid.app.Activity;
importandroid.app.AlarmManager;
importandroid.app.PendingIntent;
importandroid.app.TimePickerDialog;
importandroid.app.TimePickerDialog.OnTimeSetListener;
importandroid.content.Intent;
importandroid.os.Bundle;
importandroid.view.View;
importandroid.view.View.OnClickListener;
importandroid.widget.Button;
importandroid.widget.TimePicker;
importandroid.widget.Toast;
publicclassMainActivityextendsActivity {
privateButton btnSetClock;
privateButton btnbtnCloseClock;
privateAlarmManager alarmManager;
privatePendingIntent pi;
@Override
protectedvoidonCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btnSetClock = (Button) findViewById(R.id.btnSetClock);
btnbtnCloseClock = (Button) findViewById(R.id.btnCloseClock);
// ①获取AlarmManager对象:
alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);
// 指定要启动的是Activity组件,通过PendingIntent调用getActivity来设置
Intent intent = newIntent(MainActivity.this, ClockActivity.class);
pi = PendingIntent.getActivity(MainActivity.this, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT);
btnSetClock.setOnClickListener(newOnClickListener() {
@Override
publicvoidonClick(View v) {
Calendar currentTime = Calendar.getInstance();
// 弹出一个时间设置的对话框,供用户选择时间
newTimePickerDialog(MainActivity.this,0,
newOnTimeSetListener() {
@Override
publicvoidonTimeSet(TimePicker view,
inthourOfDay,intminute) {
//设置当前时间
Calendar c = Calendar.getInstance();
c.setTimeInMillis(System.currentTimeMillis());
// 根据用户选择的时间来设置Calendar对象
c.set(Calendar.HOUR, hourOfDay);
c.set(Calendar.MINUTE, minute);
// ②设置AlarmManager在Calendar对应的时间启动Activity
alarmManager.set(AlarmManager.RTC_WAKEUP,
c.getTimeInMillis(), pi);
// 提示闹钟设置完毕:
Toast.makeText(MainActivity.this, 闹钟设置完毕~,
Toast.LENGTH_SHORT).show();
}
}, currentTime.get(Calendar.HOUR_OF_DAY), currentTime
.get(Calendar.MINUTE),false).show();
btnbtnCloseClock.setVisibility(View.VISIBLE);
}
});
btnbtnCloseClock.setOnClickListener(newOnClickListener() {
@Override
publicvoidonClick(View v) {
alarmManager.cancel(pi);
btnbtnCloseClock.setVisibility(View.GONE);
Toast.makeText(MainActivity.this, 闹钟已取消, Toast.LENGTH_SHORT)
.show();
}
});
}
}
ClockActivity.java:
packagecom.jay.example.alarmmanagerdemo;
importandroid.app.Activity;
importandroid.app.AlertDialog;
importandroid.content.DialogInterface;
importandroid.content.DialogInterface.OnClickListener;
importandroid.media.MediaPlayer;
importandroid.os.Bundle;
publicclassClockActivityextendsActivity {
privateMediaPlayer mediaPlayer;
@Override
protectedvoidonCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_clock);
mediaPlayer = MediaPlayer.create(this, R.raw.pig);
//mediaPlayer.setLooping(true);
mediaPlayer.start();
//创建一个闹钟提醒的对话框,点击确定关闭铃声与页面
newAlertDialog.Builder(ClockActivity.this).setTitle(闹钟).setMessage(小猪小猪快起床~)
.setPositiveButton(关闭闹铃,newOnClickListener() {
@Override
publicvoidonClick(DialogInterface dialog, intwhich) {
mediaPlayer.stop();
ClockActivity.this.finish();
}
}).show();
}
}
!!!另外别忘了需要在AndroidManifest.xml对ClockActivity进行注册哦!
![\](http://www.2cto.com/uploadfile/Collfiles/20150304/2015030409361975.png)
总结核心流程吧:
①AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);
获得系统提供的AlarmManager服务的对象
②Intent设置要启动的组件:
Intent intent = new Intent(MainActivity.this, ClockActivity.class);
③PendingIntent对象设置动作,启动的是Activity还是Service,又或者是广播!
PendingIntent pi = PendingIntent.getActivity(MainActivity.this, 0, intent, 0);
④调用AlarmManager的set( )方法设置单次闹钟的闹钟类型,启动时间以及PendingIntent对象!
alarmManager.set(AlarmManager.RTC_WAKEUP,c.getTimeInMillis(), pi);
注1: 博文源地址:http://www.2cto.com/kf/201503/380190.html
注2:在原博文基础上有各关键知识点详细说明,引用了其他资料。原博文有一处问题: MainActivity.java:c.set(Calendar.HOUR, hourOfDay)和下面的<span style="line-height: 15.3999996185303px; font-family: 宋体;">currentTime.get(Calendar.HOUR_OF_DAY)其中“</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">Calendar.HOUR</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">”和</span>
<span style="line-height: 15.3999996185303px; font-family: 宋体;">“</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">Calendar.HOUR_OF_DAY</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">”必须是一个参数,必须一致</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">;“</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">Calendar.HOUR_OF_DAY</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">”是24小时,</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">“</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">Calendar.HOUR</span><span style="line-height: 15.3999996185303px; font-family: 宋体;">”</span>
<span style="line-height: 15.3999996185303px; font-family: 宋体;">是12小时制。导致的结果是上午可以正常运行程序,下午运行程序无反应(尼玛我特么找了两天才搞明白这么回事)。</span>
<span style="line-height: 15.3999996185303px; font-family: 宋体;">
</span>
。