- 浏览: 760663 次
- 性别:
- 来自: 成都
文章分类
- 全部博客 (139)
- 玩转Android (48)
- Android创意美工 (0)
- Android杂谈 (23)
- Android实例练习 (2)
- Android ROM研究 (5)
- Android NDK开发指南 (5)
- Android NDK (0)
- Android Tips (3)
- Windows Phone 7 (5)
- iPhone (0)
- HTML5学习室 (0)
- JAVA (9)
- SSH+ibatis (8)
- PHP (0)
- IT生活 (1)
- linux (2)
- C (4)
- C++ (1)
- web 前端 (1)
- 云计算 (0)
- 设计模式 (0)
- C# (2)
- 其他 (1)
- 数据结构 (5)
- Web开发 (10)
- 数据库 (3)
- 搜索引擎 (0)
- Go语言 (0)
最新评论
-
wi100sh:
多谢分享~
玩转Android---UI篇---ImageButton(带图标的按钮) -
zhanghaichang:
好文章的。
高性能web开发技术(一) -
yingang:
引用classes.dex.dex2jar.jar 拖入 j ...
Andorid杂谈---Apk文件的反编译 -
扶摇诺:
讲解的简明易懂,多谢啦!
玩转Android---UI篇---LinearLayout(线性布局) -
a13429921973:
更为详细的图文介绍,可参考这个http://blog.csdn ...
Android ROM研究---CyanogenMod源代码下载及编译
闹钟已经学过一段时间了,但是对它了解的不是很多,由于最近开发的一个小应用会用到这个功能,所以重新学习了一下,以便能在以后忘记的时候记起来,也方便其他人学习
实现闹钟有很多中方式,比如可以使用Handler+Timer(需依赖应用程序生命周期),AlarmManager等,而我们需要时间服务不依赖应用程序而存在,即应用程序启动服务,但是即使关闭应用程序,时间服务依然运行,这就需要使用AlarmManager了
首先需要了解一下闹钟需要用得到的知识点
1、实现闹钟需要用到AlarmManager来实现,这个类实现系统警告服务,可以设定一个时间来完成指定的事情,只要在程序中设置了警报服务,就可以通过调用onReceive()方法执行你要做的事情,即使是待机状态,也不会影响运行。
可以通过Context.getSystemService()方法来获得该服务。
获得AlarmManager对象代码
AlarmManager am = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
2、常量
int | ELAPSED_REALTIME | Alarm time in SystemClock.elapsedRealtime() (time since boot, including sleep). |
int | ELAPSED_REALTIME_WAKEUP | Alarm time in SystemClock.elapsedRealtime() (time since boot, including sleep), which will wake up the device when it goes off. |
long | INTERVAL_DAY | |
long | INTERVAL_FIFTEEN_MINUTES | Available inexact recurrence intervals recognized by setInexactRepeating(int, long, long, PendingIntent)
|
long | INTERVAL_HALF_DAY | |
long | INTERVAL_HALF_HOUR | |
long | INTERVAL_HOUR | |
int | RTC | Alarm time in System.currentTimeMillis() (wall clock time in UTC). |
int | RTC_WAKEUP | Alarm time in System.currentTimeMillis() (wall clock time in UTC), which will wake up the device when it goes off. |
AlarmManager.RTC,硬件闹钟,不唤醒手机(也可能是其它设备)休眠;当手机休眠时不发射闹钟。
AlarmManager.RTC_WAKEUP,硬件闹钟,当闹钟发躰时唤醒手机休眠;
AlarmManager.ELAPSED_REALTIME,真实时间流逝闹钟,不唤醒手机休眠;当手机休眠时不发射闹钟。
AlarmManager.ELAPSED_REALTIME_WAKEUP,真实时间流逝闹钟,当闹钟发躰时唤醒手机休眠;
RTC闹钟和ELAPSED_REALTIME最大的差别就是前者可以通过修改手机时间触发闹钟事件,后者要通过真实时间的流逝,即使在休眠状态,时间也会被计算。
3、Intent与PendingIntent的区别及含义
这里少不了用这两个东西,有点经验的都明白是个什么东西,但是他俩的具体区别确有时候搞不清楚,所幸放到一起,以便以后方便查询
Intent是意图的意思,它是一个马上执行的东西,比如Activity跳转,执行
Intent intent = new Intent();
intent.setClass(Test1.this, Test2.class);
startActivity(intent);
那么Intent会马上从Test1跳转到Test2
PendingIntent中Pending意思为即将发生的,待定。也就是不确定什么时候会发生,可能是未来的某个时间发生,可以理解为一中延迟的Intent,但是它一般会由别的程序触发。
Intent intent = new Intent(this, Test2.class);
PendingIntent pi = PendingIntent.getActivity(this, 0, intent, 0);
PendingIntent可以封装Activity,BroadcastReceiver,Service,与Intent有区别的是PendingIntent可以脱离应用程序而存在
其他参考:
http://wayfarer.iteye.com/blog/586159
http://gundumw100.iteye.com/blog/825537
http://blog.sina.com.cn/s/blog_5da93c8f0100u7pb.html
http://www.eoeandroid.com/forum.php?mod=viewthread&tid=68063
4、公共方法
public void cancel (PendingIntent operation)
取消警报,可以是任何类型(2中的那4中类型),比如取消闹钟
public void set (int type, long triggerAtTime, PendingIntent operation)
设置警报
参数:
第一个参数:One of ELAPSED_REALTIME, ELAPSED_REALTIME_WAKEUP, RTC or RTC_WAKEUP.
第二个参数:Time the alarm should go off, using the appropriate clock (depending on the alarm type).
第三个参数:Action to perform when the alarm goes off; typically comes from IntentSender.getBroadcast()
.
go off这里是睡眠的意思
public void setInexactRepeating (int type, long triggerAtTime, long interval, PendingIntent operation)
设置不精准重复周期
不精准重复周期的意思是:比如有很多类似的警报,那么当设置的类型范围比较大的时候,这些警报就会合并为一个警报,这样可以不用每次都执行警报,是一种节能型
参数的意思可以参见setRepeating部分
public void setRepeating (int type, long triggerAtTime, long interval, PendingIntent operation)
设置精准重复周期
这里有四个参数
第一个参数:即警报的类型,一般取值是AlarmManager.RTC和AlarmManager.RTC_WAKEUP,如果为RTC表示为一个正常的定时器,如果是RTC_WAKEUP则除了有定时器外还可以有震动或者响铃。另外就是RTC在手机睡眠的时候不发射警报,而RTC_WAKEUP则在睡眠的时候也会发射警报
第二个参数:第一次运行时要等待的时间,也就是执行延迟时间,单位是毫秒。这个参数有些难理解,有问题欢迎奥手们纠正
我的理解:
假如设置为System.currentTimeMillis()+(10*1000),表示系统会在设置警报10秒后第一次发送警报,也就是第一次响铃。这样警报服务会在10秒后进入休眠状态,但是它依赖于第一个参数的类型
英文原意:Time the alarm should first go off, using the appropriate clock (depending on the alarm type).
第三个参数:表示执行的时间间隔,单位是毫秒,也就是每过多久发射一次警报,一般都是以天为单位
第四个参数:一个PendingIntent对象,即到时间后要执行的操作
public void setTime (long millis)
设置系统时间,需要权限(android.permission.SET_TIME)
public void setTimeZone (String timeZone)
设置系统默认时区,需要权限(android.permission.SET_TIME_ZONE)
一、一个闹钟例子
这个例子在网上都可以查得到,但是具体来源已经找不到了,所以就直接采用了
这里需要注意的是TimePickerDialog中的五个参数含义,可以参考:
http://blog.csdn.net/yang_hui1986527/article/details/6839342
先看图
下面的是第一次执行时,由于设置的第一延迟执行时间是System.currentTimeMillis(),所以会马上执行
下面的是过了一分钟后执行的效果,因为设置的周期是1分钟,不过一般设置的周期都是以天数为准的
源代码
package com.loulijun.demo2; import java.util.Calendar; import android.app.Activity; import android.app.AlarmManager; import android.app.PendingIntent; import android.app.TimePickerDialog; import android.content.Intent; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.TextView; import android.widget.TimePicker; public class Demo2Activity extends Activity { private TextView tv = null; private Button setTime,cancelTime; private Calendar c = null; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); tv = (TextView)findViewById(R.id.tv); setTime = (Button)findViewById(R.id.setAlarm); cancelTime = (Button)findViewById(R.id.cancelAlarm); //得到日历实例,主要是为了下面的获取时间 c = Calendar.getInstance(); setTime.setOnClickListener(new Button.OnClickListener() { @Override public void onClick(View arg0) { c.setTimeInMillis(System.currentTimeMillis()); int hour = c.get(Calendar.HOUR_OF_DAY); int minute = c.get(Calendar.MINUTE); new TimePickerDialog(Demo2Activity.this, minute, new TimePickerDialog.OnTimeSetListener() { @Override public void onTimeSet(TimePicker view, int hourOfDay, int minute) { //是设置日历的时间,主要是让日历的年月日和当前同步 c.setTimeInMillis(System.currentTimeMillis()); //设置小时分钟,秒和毫秒都设置为0 c.set(Calendar.HOUR_OF_DAY, hourOfDay); c.set(Calendar.MINUTE, minute); c.set(Calendar.SECOND, 0); c.set(Calendar.MILLISECOND, 0); Intent intent = new Intent(Demo2Activity.this, AlarmReceiver.class); PendingIntent pi = PendingIntent.getBroadcast(Demo2Activity.this, 0, intent, 0); //得到AlarmManager实例 AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE); //根据当前时间预设一个警报 am.set(AlarmManager.RTC_WAKEUP, c.getTimeInMillis(), pi); /** * 第一个参数是警报类型;第二个参数是第一次执行的延迟时间,可以延迟,也可以马上执行;第三个参数是重复周期为一天 * 这句话的意思是设置闹铃重复周期,也就是执行警报的间隔时间 */ // am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+(60*1000), // (24*60*60*1000), pi); am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+(10*1000), 30000, pi); String msg = "设置闹钟时间为"+format(hourOfDay)+":"+format(minute); tv.setText(msg); } }, hour, minute, true).show(); //上面的TimePickerDialog中的5个参数参考:http://blog.csdn.net/yang_hui1986527/article/details/6839342 } }); cancelTime.setOnClickListener(new Button.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(Demo2Activity.this, AlarmReceiver.class); PendingIntent pi = PendingIntent.getBroadcast(Demo2Activity.this, 0, intent, 0); AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE); //取消警报 am.cancel(pi); tv.setText("闹钟取消"); } }); } private String format(int x) { String s = ""+x; if(s.length() == 1) s = "0"+s; return s; } }
广播服务:
package com.loulijun.demo2; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.media.MediaPlayer; import android.widget.Toast; public class AlarmReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent arg1) { Toast.makeText(context, "闹钟时间到", Toast.LENGTH_SHORT).show(); } }
这里需要在AndroidManifest.xml中注册广播
<receiver android:name=".AlarmReceiver" android:process=":remote"/>
---------------------------华丽的分割线--------------------------------------------------------------------
下面再看一个例子,其实这个例子是基于上面修改的,增加了用SharedPreferences来保存时间参数,如果到了时间则播放音乐,播放音乐是在一个Service中,用户可以在取消闹铃的时候取消音乐
源代码
package com.loulijun.demo2; import java.util.Calendar; import android.app.Activity; import android.app.AlarmManager; import android.app.PendingIntent; import android.app.TimePickerDialog; import android.content.Intent; import android.content.SharedPreferences; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.TextView; import android.widget.TimePicker; public class Demo2Activity extends Activity { private TextView tv = null; private Button setTime,cancelTime; private Calendar c = null; private SharedPreferences sp; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); tv = (TextView)findViewById(R.id.tv); setTime = (Button)findViewById(R.id.setAlarm); cancelTime = (Button)findViewById(R.id.cancelAlarm); //该配置文件用于存放当前设置的闹钟时间信息 sp = getSharedPreferences("alarm_record", Activity.MODE_PRIVATE); //得到日历实例,主要是为了下面的获取时间 c = Calendar.getInstance(); setTime.setOnClickListener(new Button.OnClickListener() { @Override public void onClick(View arg0) { c.setTimeInMillis(System.currentTimeMillis()); int hour = c.get(Calendar.HOUR_OF_DAY); int minute = c.get(Calendar.MINUTE); new TimePickerDialog(Demo2Activity.this, minute, new TimePickerDialog.OnTimeSetListener() { @Override public void onTimeSet(TimePicker view, int hourOfDay, int minute) { //是设置日历的时间,主要是让日历的年月日和当前同步 c.setTimeInMillis(System.currentTimeMillis()); //设置小时分钟,秒和毫秒都设置为0 c.set(Calendar.HOUR_OF_DAY, hourOfDay); c.set(Calendar.MINUTE, minute); c.set(Calendar.SECOND, 0); c.set(Calendar.MILLISECOND, 0); Intent intent = new Intent(Demo2Activity.this, AlarmReceiver.class); PendingIntent pi = PendingIntent.getBroadcast(Demo2Activity.this, 0, intent, 0); //得到AlarmManager实例 AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE); //根据当前时间预设一个警报 am.set(AlarmManager.RTC_WAKEUP, c.getTimeInMillis(), pi); /** * 第一个参数是警报类型;第二个参数是第一次执行的延迟时间,可以延迟,也可以马上执行;第三个参数是重复周期为一天 * 这句话的意思是设置闹铃重复周期,也就是执行警报的间隔时间 */ // am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+(60*1000), // (24*60*60*1000), pi); am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+(10*1000), 30000, pi); String msg = hourOfDay+":"+minute; tv.setText("当前设置的闹钟时间:"+msg); sp.edit().putString(msg, msg).commit(); } }, hour, minute, true).show(); //上面的TimePickerDialog中的5个参数参考:http://blog.csdn.net/yang_hui1986527/article/details/6839342 } }); cancelTime.setOnClickListener(new Button.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(Demo2Activity.this, AlarmReceiver.class); PendingIntent pi = PendingIntent.getBroadcast(Demo2Activity.this, 0, intent, 0); AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE); //取消警报 am.cancel(pi); tv.setText("闹钟取消"); //取消闹钟的同时取消音乐 stopService(new Intent("com.loulijun.demo2.MUSIC")); } }); } }
然后是广播接收器
package com.loulijun.demo2; import java.util.Calendar; import android.app.Activity; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.widget.Toast; public class AlarmReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent arg1) { //获取配置文件中的信息,如果相同则播放音乐 SharedPreferences sp = context.getSharedPreferences("alarm_record", Activity.MODE_PRIVATE); String hour = String.valueOf(Calendar.getInstance().get(Calendar.HOUR_OF_DAY)); String minute = String.valueOf(Calendar.getInstance().get(Calendar.MINUTE)); String time = sp.getString(hour+":"+minute, null); if(time!=null) { Toast.makeText(context, "闹钟时间到", Toast.LENGTH_SHORT).show(); //启动Service播放音乐 context.startService(new Intent("com.loulijun.demo2.MUSIC")); } } }
然后是播放音乐的服务
package com.loulijun.demo2; import android.app.Service; import android.content.Intent; import android.media.MediaPlayer; import android.os.IBinder; public class MusicService extends Service { private MediaPlayer player; @Override public IBinder onBind(Intent intent) { // TODO Auto-generated method stub return null; } public void onStart(Intent intent, int startId) { super.onStart(intent, startId); player = MediaPlayer.create(this, R.raw.test); player.start(); } public void onDestroy() { super.onDestroy(); player.stop(); } }
需要在AndroidManifest.xml中增加如下的信息,一个是注册广播,一个是注册服务
<receiver android:name=".AlarmReceiver" android:process=":remote"/> <service android:name=".MusicService"> <intent-filter> <action android:name="com.loulijun.demo2.MUSIC"> </action> <category android:name="android.intent.category.default"/> </intent-filter> </service>
其他参考:
http://www.cnblogs.com/tara/archive/2011/06/09/2076043.html
开机启动Service:http://www.eoeandroid.com/forum.php?mod=viewthread&tid=8195
http://www.eoeandroid.com/forum.php?mod=viewthread&tid=109957
AlarmUI到内核执行流程:http://www.cnblogs.com/Hwangroid/archive/2011/10/13.html
http://our2848884.blog.163.com/blog/static/14685483420114225055804/
发表评论
-
Android版本自动飞行模式小蜜
2012-05-31 15:21 527可以定时切换飞行模式的一款小工具,喜欢的下! 自动飞行模式小 ... -
Android杂谈--关于解析包时错误解决方案
2012-03-21 17:17 6097最近在一个深度定制的产品上运行apk软件,由于产品所使用的系统 ... -
Android杂谈--代码混淆及proguard一些错误处理
2011-12-12 17:01 4116代码混淆说简单的其实就是为了防止别人反编译你的源代码,因为JA ... -
Android杂谈--闹钟详谈
2011-12-10 22:23 2闹钟已经学过一段时间了,但是对它了解的不是很多,由于最近开发的 ... -
Android内存泄露调试分享
2011-10-21 22:34 3233各位兄弟姐妹,Java开发中的内存泄露的问题经常会给 ... -
Android杂谈--开发游戏时选择游戏级别
2011-09-01 13:23 1475在开发游戏的时候,开始游戏时需要几个难度选择的选项供用户选择。 ... -
Android杂谈--Android生命周期
2011-09-01 10:48 2625引言 应用程序组件 ... -
Android杂谈--获取系统及应用程序(PackageManager)
2011-08-27 19:39 5670PackageManager是个非常好的东西,其他的详细的细节 ... -
Android杂谈---用MD5处理明文密码
2011-08-16 21:13 3211很多的网络相关的软件都需要用户名密码登录,在开发的时候像这些密 ... -
Androidz杂谈---在ContextMenu中添加/删除ListView的Item
2011-08-16 16:28 3038我们在写Android程序的时候,免不了要使用ListView ... -
Android杂谈---各种Toast
2011-07-30 21:58 1623相信各位对这个Toast已经了解的差不多了,不过我们还可以定义 ... -
Android杂谈---设置模拟器壁纸
2011-07-25 20:34 1606天天看着模拟器里面的那个图片感觉很恼火,所以干脆想将其换掉,顺 ... -
Android杂谈---TextView的跑马灯效果
2011-07-24 17:45 3435下面的是转载自农民伯伯的文章,但是有些属性还不是很清楚,所以又 ... -
Android杂谈---layout_x与layout_y的正确使用
2011-05-26 22:22 3095<?xml version="1.0" ... -
Android杂谈---分享eoeAndroid第1--16期资源
2011-05-26 22:00 2150Android杂谈---分享eoeAndroid第1--16期 ... -
Android杂谈---关于drawable文件夹的错误
2011-05-26 13:13 5033今天做一个东西的时候,用PS做了几个图片,但是更改了后缀,于是 ... -
Android杂谈---获取手机屏幕大小
2011-05-20 23:04 1636开发手机应用程序的时候,除了底层对API的掌握外,最重要的仍是 ... -
Android杂谈---TextView的12种文字颜色
2011-05-20 22:35 6132要想设置Android的TextView控件不同的颜色有两种方 ... -
Andorid杂谈---Apk文件的反编译
2011-04-09 14:53 38471、 首先是将下载到 ... -
Android杂谈---带图片的Toast
2011-03-24 21:28 2207当需要提示的时候,我们可以用Toast来显示信息 如: T ...
相关推荐
这篇“Android杂谈---Nexus S的ROOT教程”显然是针对想要获取Nexus S手机ROOT权限的用户,提供了一条详细的操作路径。Nexus S是由Google与三星联合推出的旗舰设备,因其开源特性而深受开发者喜爱。 首先,我们需要...
计算机汇编杂谈-理解其中的原理
标题中的“杂谈----1 显示横向滚动条”暗示了我们将会探讨的是关于在界面设计中如何实现或处理横向滚动条的技术问题。这通常涉及到前端开发,特别是网页或应用程序的用户界面(UI)部分,其中可能包括HTML、CSS和...
"项目管理故事2-实施策略杂谈-开元项目实施" 本文是基于赵磊项目经理的项目管理故事,分享实施成功经验。赵磊担任过多个项目经理职位,负责开元旅业NC项目财务实施、苏州好孩子NC财务项目实施、合肥公交NC财务第一...
「安全管理」Linux_HIDS杂谈 - WEB应用防火墙 安全管理 安全资讯 安全开发 零信任 身份管理
在"swing开发杂谈--初版本程序源码"中,可能包含了上述部分或全部知识点的实际应用,通过分析`netHelper`这个子文件夹,我们可以看到可能的网络辅助类或其他功能模块的实现。这个源码可能会演示如何使用Swing构建一...
### 软件工程历史概览 #### 一、程序设计的起源与始祖 - **十七世纪的计算器发展**:十七世纪初,人类开始尝试制造计算器,这标志着早期计算机科学的萌芽。1623年,德国的博学者Wilhelm Schickard制造了世界上第一...
这份“程序设计经验杂谈”文档,据说是出自一位大师之手,无疑为我们提供了宝贵的洞见。在本文中,我们将探讨几个关键的程序设计经验,希望能对你在编程旅程中提供一些帮助。 首先,我们来谈谈问题解决策略。在编程...
### 嵌入式开发杂谈—薛立功经典之作:关键知识点解析 #### 一、引言 在嵌入式开发领域,《嵌入式开发杂谈》是薛立功先生的经典著作之一,该作品旨在解答新手在入门阶段遇到的各种问题。通过作者的经验分享,帮助...
在嵌入式开发领域,FPGA(Field Programmable Gate Array)和DSP(Digital Signal Processor)是两种非常重要的处理器,它们各自具有独特的特性和优势。FPGA是一种可编程的硬件平台,能够根据用户的需求配置出不同的...
在嵌入式开发领域,FPGA(Field Programmable Gate Array)和DSP(Digital Signal Processor)是两种重要的处理器类型,它们各有特色,广泛应用于各种复杂的数字信号处理任务。本文将探讨这两种技术的核心特点,以及...
【Java动态加载机制】 Java的动态加载机制是其设计的一大亮点,它允许程序在运行时按需加载类,而不是一次性加载所有类。这种机制降低了内存消耗,并提高了程序的灵活性。当我们编写Java程序并运行时,只有当我们...
在本文中,作者分享了作为一名计算机专业学生的编程经验,主要关注Java语言。文章分为三个部分,分别讨论了Java的动态加载机制、查找class文件的原理以及JDK和JRE的区别。 1. 动态加载机制: Java的动态加载机制...
为了解学术界东西方思维习惯的演变,本文建立了以随机概率论为基础的数学模型,考虑东西价值观差异,以揭示政府管制和道德引导在学术界树立正确价值观和文化自信中的影响以及必要性。数学模型都不是完全真实的,需要...