在前两篇文章中,我们了解了SMSPopup的概况 及研究了目录结构和Mainfest文件 ,现在我们深入代码,来看看SMSPopup的具体实现。
SMSPopupConfigActivity
此类为程序入口,主要是SMSPopup的相关配置,该类继承了PreferenceActivity,该类是使用SharedPreferneces以键值对的形式保存相关配置数据的。
在onCreate方法中,addPreferencesFromResource(R.xml.preferences);作用是从preferences.xml中加载PreferenceActivity的UI。
preferences.xml中的元素有:
PreferenceScreen --xml顶层节点,若出现在子节点中,当点击此Preference时,会另外显示一屏
PreferenceCategory --代表一类相关的配置,用于分组
以下四类为具体的Preference,依次是单选框,输入编辑框,列表选择,铃声选择。
CheckBoxPreference
EditTextPreference
ListPreference
RingtonePreference
程序就是通过这些基本的Preference来显示及更改配置信息的,也可以自定义Preference,定义类,继承上面几种Preference中的一种,在xml文件中引用即可,net.everythingandroid.smspopup.preferences包中的5个类都是自定义的Preference。
SMSPopupUtilsService
此类中利用了Android Message机制来使不同线程间通信。分析之前建议先了解下Android线程相关知识:http://zjf1428.iteye.com/blog/695491 ,文中详细描述了怎样使用Message、Handle、Looper相关类来完成线程间的通信。
Android程序在启动时,系统会启动一个对应的主线程,主线程主要负责处理与UI相关的事件,Android UI操作并不是线程安全的并且这些操作必须在主线程中执行,这就引发了一个问题,当主线程正在做一些比较耗时的操作的时候,如正从网络上下载一个大图片,或者访问数据库,由于主线程被这些耗时的操作阻塞住,无法及时的响应用户的事件,从用户的角度看会觉得程序已经死掉,为增强用户体验,Android设置了一个5秒钟超时时间:一旦用户的事件由于主线程阻塞而超过5秒 钟没有响应,Android会 弹出一个应用程序没有响应的对话框。作为开发者,应当尽量避免在主线程中执行比较耗时的操作。来看看此类中是怎样做的:
@Override public void onCreate() { HandlerThread thread = new HandlerThread(Log.LOGTAG, Process.THREAD_PRIORITY_BACKGROUND); thread.start(); context = getApplicationContext(); mServiceLooper = thread.getLooper(); mServiceHandler = new ServiceHandler(mServiceLooper); } @Override public void onStart(Intent intent, int startId) { //mResultCode = intent.getIntExtra("result", 0); Message msg = mServiceHandler.obtainMessage(); msg.arg1 = startId; msg.obj = intent; mServiceHandler.sendMessage(msg); }
此类继承了Service(后台服务),在onCreate()方法中,首先定义了一个HandlerThread,为ServiceHandler的创建准备Looper,Looper是MessageQueue和Handler之间的桥梁,具体参见上面的文章。再看onStart()方法,构造了一条Message,包含了一个intent对象,改intent来自于SMSPopupUtils调用startService(intent),intent中包含了一条SmsMmsMessage(短信实体)和Action(标志将短信设为已读的动作)。最后一句mServiceHandler.sendMessage(msg)执行完成后,Message就被转到Handler处理
private final class ServiceHandler extends Handler { public ServiceHandler(Looper looper) { super(looper); } @Override public void handleMessage(Message msg) { Log.v("SMSPopupUtilsService: handleMessage()"); int serviceId = msg.arg1; Intent intent = (Intent) msg.obj; String action = intent.getAction(); if (ACTION_MARK_THREAD_READ.equals(action)) { Log.v("SMSPopupUtilsService: marking thread read"); SmsMmsMessage message = new SmsMmsMessage( context, intent.getExtras()); message.setThreadRead(); } else if (ACTION_OTHER.equals(action)) { } finishStartingService(SMSPopupUtilsService.this, serviceId); } }
Handler类只有两个方法,一个构造方法,一个重写的handleMessage,用于处理sendMessage()发过来的消息,在handleMessage方法中,首先取出Message中包含的数据,根据Aciton做处理,此处Action表示将短信设为已读,由message.setThreadRead()完成。最后调用finishStartingService()停止Service并释放唤醒锁(释放CPU,使屏幕变暗)。
实际上,在创建需要与用户界面交互的长时间运行的任务时,Android为我们提供了更简单的方法---使用AsyncTask类,具体用法见上面的文章,接下来动手改造SMSPopupUtilsService类,我们用简便的AsyncTask来替代Handler,改造过程比较顺利,就遇到一个把方法参数写错的小错误,改造后的代码确实比原来的简洁干净些。
代码学习过程中,发现一处很有意思的地方:
if (ACTION_MARK_THREAD_READ.equals(action)) { if (Log.DEBUG) Log.v("SMSPopupUtilsService: Marking thread read"); SmsMmsMessage message = new SmsMmsMessage(context, intent.getExtras()); message.setThreadRead(); } else if (ACTION_MARK_MESSAGE_READ.equals(action)) { if (Log.DEBUG) Log.v("SMSPopupUtilsService: Marking message read"); SmsMmsMessage message = new SmsMmsMessage(context, intent.getExtras()); message.setMessageRead(); } else if (ACTION_DELETE_MESSAGE.equals(action)) { if (Log.DEBUG) Log.v("SMSPopupUtilsService: Deleting message"); SmsMmsMessage message = new SmsMmsMessage(context, intent.getExtras()); message.delete(); } else if (ACTION_QUICKREPLY.equals(action)) { if (Log.DEBUG) Log.v("SMSPopupUtilsService: Quick Reply to message"); SmsMmsMessage message = new SmsMmsMessage(context, intent.getExtras()); message.replyToMessage(intent.getStringExtra( SmsMmsMessage.EXTRAS_QUICKREPLY)); } else if (ACTION_UPDATE_NOTIFICATION.equals(action)) { if (Log.DEBUG) Log.v("SMSPopupUtilsService: Updating notification"); updateNotification(intent); }
请注意前两个判断,完全相同,也就是如果符合第一个条件,if里面的代码或执行两边,也就是将信息两次置为已读,完全多余,看了一下最新版本,代码还是如此。
实际上参看AsyncTask的源代码 ,发现原来AsyncTask的实现竟然就是使用Handler Message机制,只是把它封装得更易用些,我们还是要熟练使用Message来处理线程间通信。