Android提供的系统服务之--AlarmManager(闹钟服务)

来源:互联网 发布:移动终端软件开发技术 编辑:程序博客网 时间:2024/05/19 02:25

本节引言:

本节主要介绍的是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进行注册哦!

 \

\

总结核心流程吧:

①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>


0 0