• 1. 第11章 综合示例设计与开发
  • 2. 本章学习目标:了解Android应用程序的设计和开发过程 掌握使用多种组件进行Android程序开发的方法
  • 3. 11.1 需求分析 设计本章的初衷 希望读者能够根据实际项目的需求,准确的分析出Android应用程序开发所可能涉及到的知识点,并学会如何通过分析软件的需求,快速的设计出应用程序的用户界面和模块结构,并最终完成应用程序的开发和调试
  • 4. 11.1 需求分析功能需求 本章提供的“天气预报短信服务软件”是一个略微复杂的示例。在这个综合示例中,有一个显示天气情况的用户界面,可以通过图片和文字显示当前和未来几天的天气状况,包括温度、湿度、风向和雨雪情况等。这些天气数据是通过后台服务获取的,这个后台服务可以按照一定时间间隔,从Google上获取天气预报信息,并将天气信息保存在后台服务中。示例还需要提供基于SMS短信的天气数据服务,其他手机用户可以向本示例所在的手机上发送SMS短信,并在短信中包含用户指定的关键字,则可以将保存在后台服务中的天气情况,再通过SMS短信回复给用户。最后,每个被发送的SMS短信都要被记录下来,用户可以浏览或删除这些回复信息
  • 5. 11.1 需求分析界面需求 本示例包含三个主要的用户界面 显示天气预报的用户界面 显示已发送SMS短信的用户界面 浏览和设置配置信息的用户界面
  • 6. 11.1 需求分析内部功能 隐藏在用户界面后面的内部功能,是用户界面能够正确实现的基础 显示天气预报的用户界面 获取Google的天气数据 显示SMS短信的用户界面 根据关键字监视SMS短信 发送包含天气信息的SMS短信 将发送的SMS短信写入数据库 浏览和设置配置信息的用户界面 将用户设置的配置信息保存到数据库 启动时读取数据库中的配置信息 恢复缺省设置
  • 7. 11.2 程序设计11.2.1 用户界面设计 详细分析应用程序中三个主要用户界面包含的显示内容 在“显示天气预报的用户界面”中,根据Google可以提供的数据,在界面上可以显示当前的天气状况,包括城市名称、温度、湿度、风向、雨雪情况和获取数据时间等信息;还可以显示未来四天的天气状况,但仅包括温度和雨雪情况 在“显示已发送SMS短信的用户界面”中,应显示每个回复短信的时间、目标手机号码、城市名称、当天的天气状况和未来一天的天气状况 在“浏览和设置配置信息的用户界面”中,应显示获取天气预报的目标城市名称、获取数据的频率和短信的关键字,并允许用户设置是否提供短信服务以及是否记录回复的短信信息
  • 8. 11.2 程序设计11.2.1 用户界面设计 用户界面的草图
  • 9. 11.2 程序设计11.2.2 数据库设计 本示例主要存储两种数据 配置信息:因为配置信息的数据量很小,从Android支持的存储方式上分析,可以保存在SharePreference、文件或SQLite数据库中 SMS短信服务信息: SMS短信服务信息是一个随着时间推移而不断增加的数据,属于文本信息,而且有固定的格式,因此适合使用SQLite数据库进行存储 综合分析这两种需要存储的数据,选择SQLite数据库作为存储数据的方法
  • 10. 11.2 程序设计11.2.2 数据库设计 配置信息 配置信息中主要保存天气信息查询的城市名称,访问Google更新天气信息的频率,请求天气信息SMS短信的关键字,以及是否提供短信服务和是否记录短信服务内容 配置信息的数据库表结构 属性数据类型说明_idinteger自动增加的主键city_nametext进行天气信息查询的城市名refresh_speedtext进行天气信息查询的频率,单位为秒/次sms_servicetext是否提供短信服务,即接收到请求短信后是否回复包含天气信息的短信sms_infotext是否记录发出的SMS短信的信息key_wordtext短信服务的关键字,用以确定哪条短信是请求天气服务的短信
  • 11. 11.2 程序设计11.2.2 数据库设计 SMS短信服务信息 SMS短信服务信息主要保存请求服务短信的发送者、短信内容、接收时间和回复信息的内容 SMS短信服务信息的数据库表结构属性数据类型说明_idinteger自动增加的主键sms_sendertext请求服务短信的发送者sms_bodytext请求服务短信的内容信息sms_receive_timetext接收到请求服务短信的时间return_resulttext回复短信的内容
  • 12. 11.2 程序设计11.2.3 程序模块设计 从功能需求上分析,可以将整个应用程序划分为4个模块,分别是用户界面、后台服务、数据库适配器和短信监听器。下图是模块结构图
  • 13. 11.2 程序设计11.2.3 程序模块设计 由模块结构图中可知,后台服务是整个应用程序的核心,主要包含两个子模块,一个是“数据获取模块”,负责周期性的从Google获取天气信息;另一个是“短信服务模块”,负责处理接收到的服务请求短信,并发送包含天气信息的短信 后台服务由用户界面通过Intent启动,启动后的后台服务可以在用户界面关闭后仍然保持运行状态,直到用户通过用户界面发送Intent停止服务,或系统因资源不足而强行关闭服务
  • 14. 11.2 程序设计11.2.3 程序模块设计 用户界面从后台服务获取天气信息,而没有直接通过网络访问Google的天气数据 一方面是因为后台服务使用了工作线程,通过后台服务获取天气数据可以避免因网络通信不畅造成界面失去响应 另一方面,在用户关闭界面后,后台服务仍然需要更新天气信息,以保证短信服务数据的准确性。用户界面还会调用数据库适配器,向SQLite数据库中写入、读取配置信息,或对SMS短信服务信息进行操作
  • 15. 11.2 程序设计11.2.3 程序模块设计 短信监听器是一个BroadcastReceiver,监视所有接收到的短信 如果短信中包含用户自定义的关键字,短信监听器则会认为这条短信是天气服务请求短信,将短信的相关信息写入后台服务的短信服务队列 如果用户在配置信息中选择无需提供短信服务,短信监听器仍然继续监听所有短信,只是后台服务不再允许将服务请求短信写入服务队列 数据库适配器封装了所有对SQLite数据库操作的方法,用户界面和后台服务会调用它实现数据库操作
  • 16. 11.3 程序开发 11.3.1 文件结构与用途 在程序开发阶段,首先确定“天气预报短信服务软件”的工程名称为WeatherDemo,包名称为edu.hrbeu.WeatherDemo,据程序模块设计的内容,建立WeatherDemo示例
  • 17. 11.3 程序开发11.3.1 文件结构与用途 WeatherDemo示例源代码的文件结构
  • 18. 11.3 程序开发11.3.1 文件结构与用途 WeatherDemo示例设置了多个命名空间,分别用来保存用户界面、数据库、后台服务、SMS短信和天气数据的源代码文件 WeatherDemo示例的命名空间命名空间说明edu.hrbeu.WeatherDemo存放与用户界面相关的源代码文件edu.hrbeu.WeatherDemo.DB存放与SQLite数据库相关的源代码文件edu.hrbeu.WeatherDemo.Service存放与后台服务相关的源代码文件edu.hrbeu.WeatherDemo.SMS 存放与SMS短信相关的源代码文件edu.hrbeu.WeatherDemo.Weather 存放与天气数据有关的源代码文件
  • 19. 11.3 程序开发11.3.1 文件结构与用途 WeatherDemo示例将不同用途的源代码文件放置在不同的命名空间中包名称文件名说明.WeatherDemoHistoryActivity.java“历史数据”页的ActivitySetupActivity.java“系统设置”页的ActivityWeatherActivity.java“天气预报”页的ActivityWeatherDemo.java程序启动缺省的Activity.WeatherDemo.DBConfig.java保存配置信息的类DBAdapter.java数据库适配器.WeatherDemo.ServiceSmsReceiver.java短信监听器WeatherAdapter.java数据获取模块WeatherService.java后台服务.WeatherDemo.SMS SimpleSms.java简化的SMS短信类SmsAdapter.java短信发送模块.WeatherDemo.WeatherForecast.java未来天气信息的类Weather.java当前天气信息的类
  • 20. 11.3 程序开发11.3.1 文件结构与用途 Android的资源文件保存在/res的子目录中 /res/drawable目录中保存的是图像文件 /res/layout目录中保存的是布局文件 /res/values目录中保存的是用来定义字符串和颜色的文件 /res/xml目录保存的是XML格式的数据文件 所有在程序开发阶段可以被调用的资源都保存在这些目录中
  • 21. 11.3 程序开发11.3.1 文件结构与用途 资源文件名称与用途 资源目录文件说明drawableicon.png图标文件sunny.png调试用的天气图片tab_history.pngTabHost中“历史数据”页的图片tab_setup.pngTabHost中“系统设置”页的图片tab_weather.pngTabHost中“天气预报”页的图片layoutdata_row.xml“历史数据”页ListActivity的每行数据的布局tab_history.xmlTabHost中“历史数据”页的布局tab_setup.xmlTabHost中“系统设置”页的布局tab_weather.xmlTabHost中“天气预报”页的布局valuescolor.xml保存颜色的XML文件string.xml保存字符串的XML文件xmlapi.xml从Google下载的天气数据文件。在程序运行时没有实际作用,但在开发过程中可以让读者了解数据格式
  • 22. 11.3 程序开发11.3.2 数据库适配器 数据库适配器是最底层的模块,主要用于封装用户界面和后台服务对SQLite数据库的操作,数据库适配器的核心代码主要在DBAdapter.java文件中 用户保存配置信息的类文件Config.java Config.java文件的全部代码如下 package edu.hrbeu.WeatherDemo.DB; public class Config { public static String CityName; public static String RefreshSpeed; public static String ProvideSmsService; public static String SaveSmsInfo; public static String KeyWord;
  • 23. 11.3 程序开发11.3.2 数据库适配器 从代码中不难看出,公有静态属性CityName、RefreshSpeed、ProvideSmsService、SaveSmsInfo和KeyWord,完全对应数据库中保存配置信息表的属性。在程序启动后,保存在数据库中的城市名称、更新频率、是否提供短信服务、是否保存短信信息和关键字等内容,将被写入这个Config类中,供其他模块在做逻辑判断时使用public static void LoadDefaultConfig(){ CityName = ""; RefreshSpeed = "60"; ProvideSmsService = "true"; SaveSmsInfo = "true"; KeyWord = "NY"; } }
  • 24. 11.3 程序开发11.3.2 数据库适配器 代码第10行的LoadDefaultConfig()函数,保存了程序内置的配置参数 此函数会在两个情况下被调用 用户主动选择“恢复缺省设置” 首次启动程序时,用来初始化保存配置参数的数据库 DBAdapter类与以往介绍过的数据库适配器类相似,都具有继承SQLiteOpenHelper的帮助类DBOpenHelper DBOpenHelper在建立数据库时,同时建立两个数据库表,并对保存配置信息的表进行了初始化,初始化的相关代码在第42~49行
  • 25. 11.3 程序开发11.3.2 数据库适配器 private static final String DB_NAME = "weather_app.db"; private static final String DB_TABLE_CONFIG = "setup_config"; private static final String DB_CONFIG_ID = "1"; private static final int DB_VERSION = 1; public static final String KEY_ID = "_id"; public static final String KEY_CITY_NAME = "city_name"; public static final String KEY_REFRESH_SPEED = "refresh_speed"; public static final String KEY_SMS_SERVICE = "sms_service"; public static final String KEY_SMS_INFO = "sms_info"; public static final String KEY_KEY_WORD = "key_word"; private static final String DB_TABLE_SMS = "sms_data"; public static final String KEY_SENDER = "sms_sender"; public static final String KEY_BODY = "sms_body"; public static final String KEY_RECEIVE_TIME = "sms_receive_time"; public static final String KEY_RETURN_RESULT = "return_result";
  • 26. 11.3 程序开发11.3.2 数据库适配器 /**静态Helper类,用于建立、更新和打开数据库*/ private static class DBOpenHelper extends SQLiteOpenHelper { public DBOpenHelper(Context context, String name, CursorFactory factory, int version) { super(context, name, factory, version); } private static final String DB_CREATE_CONFIG = "create table " + DB_TABLE_CONFIG + " (" + KEY_ID + " integer primary key autoincrement, " + KEY_CITY_NAME+ " text not null, " + KEY_REFRESH_SPEED+ " text," + KEY_SMS_SERVICE +" text, " + KEY_SMS_INFO + " text, " + KEY_KEY_WORD + " text);"; private static final String DB_CREATE_SMS = "create table " + DB_TABLE_SMS + " (" + KEY_ID + " integer primary key autoincrement, " + KEY_SENDER+ " text not null, " + KEY_BODY+ " text, " + KEY_RECEIVE_TIME +" text, " + KEY_RETURN_RESULT + " text);";
  • 27. 11.3 程序开发11.3.2 数据库适配器 @Override public void onCreate(SQLiteDatabase _db) { _db.execSQL(DB_CREATE_CONFIG); _db.execSQL(DB_CREATE_SMS); //初始化系统配置的数据表 Config.LoadDefaultConfig(); ContentValues newValues = new ContentValues(); newValues.put(KEY_CITY_NAME, Config.CityName); newValues.put(KEY_REFRESH_SPEED, Config.RefreshSpeed); newValues.put(KEY_SMS_SERVICE, Config.ProvideSmsService); newValues.put(KEY_SMS_INFO, Config.SaveSmsInfo); newValues.put(KEY_KEY_WORD, Config.KeyWord); _db.insert(DB_TABLE_CONFIG, null, newValues); }
  • 28. 11.3 程序开发11.3.2 数据库适配器 @Override public void onUpgrade(SQLiteDatabase _db, int _oldVersion, int _newVersion) { _db.execSQL("DROP TABLE IF EXISTS " + DB_TABLE_CONFIG); _db.execSQL("DROP TABLE IF EXISTS " + DB_CREATE_SMS); onCreate(_db); } }
  • 29. 11.3 程序开发11.3.2 数据库适配器 在DBAdapter类中,用户界面会调用SaveConfig()和LoadConfig(),从SQLite数据库中保存和读取配置信息。保存配置信息时,SaveConfig()函数会将Config类中的公有静态属性写入数据库;反之,LoadConfig()会将数据库中的配置信息写入Config类中的公有静态属性 SaveConfig()和LoadConfig()的代码如下 public void SaveConfig(){ ContentValues updateValues = new ContentValues(); updateValues.put(KEY_CITY_NAME, Config.CityName); updateValues.put(KEY_REFRESH_SPEED, Config.RefreshSpeed); updateValues.put(KEY_SMS_SERVICE, Config.ProvideSmsService); updateValues.put(KEY_SMS_INFO, Config.SaveSmsInfo); updateValues.put(KEY_KEY_WORD, Config.KeyWord);
  • 30. 11.3 程序开发11.3.2 数据库适配器 db.update(DB_TABLE_CONFIG, updateValues, KEY_ID + "=" + DB_CONFIG_ID, null); Toast.makeText(context, "系统设置保存成功", Toast.LENGTH_SHORT).show(); } public void LoadConfig() { Cursor result = db.query(DB_TABLE_CONFIG, new String[] { KEY_ID, KEY_CITY_NAME, KEY_REFRESH_SPEED,KEY_SMS_SERVICE, KEY_SMS_INFO, KEY_KEY_WORD}, KEY_ID + "=" + DB_CONFIG_ID, null, null, null, null); if (result.getCount() == 0 || !result.moveToFirst()){ return; }
  • 31. 11.3 程序开发11.3.2 数据库适配器 Config.CityName = result.getString(result.getColumnIndex(KEY_CITY_NAME)); Config.RefreshSpeed = result.getString(result.getColumnIndex(KEY_REFRESH_SPEED)); Config.ProvideSmsService=result.getString(result.getColumnIndex(KEY_SMS_SERVICE)); Config.SaveSmsInfo = result.getString(result.getColumnIndex(KEY_SMS_INFO)); Config.KeyWord = result.getString(result.getColumnIndex(KEY_KEY_WORD)); Toast.makeText(context, "系统设置读取成功", Toast.LENGTH_SHORT).show(); }
  • 32. 11.3 程序开发11.3.2 数据库适配器 另一个会调用DBAdapter类的是后台服务,即WeatherService类 后台服务主要调用SaveOneSms(SimpleSms sms)、DeleteAllSms()和GetAllSms()函数,分别用来保存SMS短信记录、删除所有SMS数据记录和获取所有SMS数据记录 在GetAllSms()函数中,调用了一个私有函数ToSimpleSms(Cursor cursor),用来将从数据库获取的数据转换为SimpleSms对象数组 SimpleSms类将在下一小节进行介绍 下面是SaveOneSms(SimpleSms sms)、DeleteAllSms()和GetAllSms()函数的代码
  • 33. 11.3 程序开发11.3.2 数据库适配器 public void SaveOneSms(SimpleSms sms){ ContentValues newValues = new ContentValues(); newValues.put(KEY_SENDER, sms.Sender); newValues.put(KEY_BODY, sms.Body); newValues.put(KEY_RECEIVE_TIME, sms.ReceiveTime); newValues.put(KEY_RETURN_RESULT, sms.ReturnResult); db.insert(DB_TABLE_SMS, null, newValues); } public long DeleteAllSms() { return db.delete(DB_TABLE_SMS, null, null); } public SimpleSms[] GetAllSms() { Cursor results = db.query(DB_TABLE_SMS, new String[] { KEY_ID, KEY_SENDER, KEY_BODY, KEY_RECEIVE_TIME, KEY_RETURN_RESULT}, null, null, null, null, null); return ToSimpleSms(results); } private SimpleSms[] ToSimpleSms(Cursor cursor){
  • 34. 11.3 程序开发11.3.2 数据库适配器 int resultCounts = cursor.getCount(); if (resultCounts == 0 || !cursor.moveToFirst()){ return null; } SimpleSms[] sms = new SimpleSms[resultCounts]; for (int i = 0 ; i
  • 35. 11.3 程序开发11.3.3 短信监听器 短信监听器本质上是BroadcastReceiver,用于监听Android系统所接收到的所有SMS短消息,可以在应用程序关闭后仍然继续运行,核心代码在SmsReceiver.java文件中 在介绍SmsReceiver类前,先说明用来保存SMS短信内容和相关信息的SimpleSms类。android.telephony.gsm.SmsMessage是Android提供的短信类,但这里需要一个更精简、小巧的类,保存少量的信息,因此构造了SimpleSms类,仅用来保存短信的发送者、内容、接收时间和返回结果。这里的“返回结果”指的是返回包含天气信息的短信内容
  • 36. 11.3 程序开发11.3.3 短信监听器 SimpleSms.java文件完整代码: package edu.hrbeu.WeatherDemo.SMS; import java.text.SimpleDateFormat; public class SimpleSms { public String Sender; public String Body; public String ReceiveTime; public String ReturnResult; public SimpleSms(){ }
  • 37. 11.3 程序开发11.3.3 短信监听器 第5行到第8行代码的属性Sender、Body、ReceiveTime和ReturnResult,分别表示SMS短信的发送者、内容、接收时间和返回结果 第15行和第16行的代码在SimpleSms类的构造函数中,直接将系统时间以“年-月-日 小时:分:秒”的格式保存在ReceiveTime属性中 public SimpleSms(String sender, String body){ this.Sender = sender; this.Body = body; SimpleDateFormat tempDate = new SimpleDateFormat("yyyy-MM-dd" + " " + "hh:mm:ss"); this.ReceiveTime = tempDate.format(new java.util.Date()); this.ReturnResult = ""; } }
  • 38. 11.3 程序开发11.3.3 短信监听器 SmsReceiver类继承BroadcastReceiver,重载了onReceive()函数 系统消息的识别和关键字的识别并不复杂,只要接收android.provider.Telephony.SMS_RECEIVED类型的系统消息,则表明是Android系统接收到了短信;将短信的内容拆分后,判断消息内容是否是配置信息中定义的关键字,即可判断该短信是否为天气服务请求短信
  • 39. 11.3 程序开发11.3.3 短信监听器 SmsReceiver.java文件的核心代码 public class SmsReceiver extends BroadcastReceiver{ private static final String SMS_ACTION = "android.provider.Telephony.SMS_RECEIVED"; @Override public void onReceive(Context context, Intent intent){ if (intent.getAction().equals(SMS_ACTION)){ Bundle bundle = intent.getExtras(); if (bundle != null){ Object[] objs = (Object[]) bundle.get("pdus"); SmsMessage[] messages = new SmsMessage[objs.length]; for (int i = 0; i
  • 40. 11.3 程序开发11.3.3 短信监听器 String smsBody = messages[0].getDisplayMessageBody(); String smsSender = messages[0].getDisplayOriginatingAddress(); if (smsBody.trim().equals(Config.KeyWord) && Config.ProvideSmsService.equals("true")){ SimpleSms simpleSms = new SimpleSms(smsSender, smsBody); WeatherService.RequerSMSService(simpleSms); Toast.makeText(context, "接收到服务请求短信", Toast.LENGTH_SHORT).show(); } } } }
  • 41. 11.3 程序开发11.3.3 短信监听器 第10行代码将带有pdus字符串特征的对象,通过Bundle.get()函数提取出来 在第12行代码使用SmsMessage.CreateFromPdu()函数构造SmsMessage对象 在第11行代码使用循环语句是因为接收到的短信可能不止一条 从第14行和第15行代码上看,这里只处理第1条短信 第17行代码构造SimpleSms对象 在代码第18行调用WeatherService类的RequerSMSService()函数,将SimpleSms对象添加到短信队列中
  • 42. 11.3 程序开发11.3.3 短信监听器 在AndroidManifest.xml文件中注册短信监听器SmsReceiver,并声明可以接收短信的用户许可android.permission.RECEIVE_SMS 如果注册的组件不在根命名空间中,则需要将子命名空间写在类的前面
  • 43. 11.3 程序开发11.3.3 短信监听器 例如下面在代码第1行中,因为SmsReceiver.java文件在edu.hrbeu.WeatherDemo.Service命名空间下,而不在根命名空间edu.hrbeu.WeatherDemo下,因此注册组件时需要在类名SmsReceiver前添加.Service
  • 44. 11.3 程序开发11.3.4 后台服务 后台服务是WeatherDemo示例的核心模块,在用户启动后持续在后台运行,直到用户手动停止服务 后台服务功能 一是发送包含天气信息的SMS短信(短信发送模块) 二是周期性的获取Google的天气数据(数据获取模块)  
  • 45. 11.3 程序开发11.3.4 后台服务 短信发送模块:后台服务在单独的线程上运行 首先调用ProcessSmsList()函数,检查短信队列中是否有需要回复的短信 然后调用GetGoogleWeatherData()函数获取天气数据 最后线程暂停1秒,以释放CPU资源 WeatherDemo示例后台服务的核心代码在WeatherService.java文件中
  • 46. 11.3 程序开发11.3.4 后台服务 下面是线程调用函数的部分代码 private static ArrayList smsList = new ArrayList(); private Runnable backgroudWork = new Runnable(){ @Override public void run() { try{ while(!Thread.interrupted()){ ProcessSmsList(); GetGoogleWeatherData(); Thread.sleep(1000); } } catch (InterruptedException e) { e.printStackTrace(); } } };
  • 47. 11.3 程序开发11.3.4 后台服务 ProcessSmsList()函数用来检查短信列表smsList,并根据Weather类中保存的天气数据,向请求者的发送回复短信 WeatherService.java文件的ProcessSmsList()函数代码如下
  • 48. 11.3 程序开发11.3.4 后台服务 private void ProcessSmsList(){ if (smsList.size()==0){ return; } SmsManager smsManager = SmsManager.getDefault(); PendingIntent mPi = PendingIntent.getBroadcast(this, 0, new Intent(), 0); while(smsList.size()>0){ SimpleSms sms = smsList.get(0); smsList.remove(0); smsManager.sendTextMessage(sms.Sender, null, Weather.GetSmsMsg(), mPi, null); sms.ReturnResult = Weather.GetSmsMsg(); SaveSmsData(sms); } }
  • 49. 11.3 程序开发11.3.4 后台服务 发送短信是使用SmsManager对象的sendTextMessage()方法,该方法一共需要5个参数 第1个参数是收件人地址 第2个参数是发件人地址 第3个参数是短信正文 第4个参数是发送服务 第5个参数是送达服务 sendTextMessage()方法的收件人地址和短信正文是不可为空的参数,而且一般GSM规范要求短信内容要控制在70个汉字以内 代码第8行的Weather.GetSmsMsg(),用来获得供回复短信使用的天气信息,因为考虑到短信的字数限制,仅返回当天和未来一天的天气状况
  • 50. 11.3 程序开发11.3.4 后台服务 Weather.java文件的代码如下 package edu.hrbeu.WeatherDemo.Weather; import android.graphics.Bitmap; public class Weather { public static String city; public static String forecase_date; public static String current_date_time; public static String current_condition; public static String current_temp; public static String current_humidity; public static String current_image_url; public static Bitmap current_image; public static String current_wind;
  • 51. 11.3 程序开发11.3.4 后台服务 public static Forecast[] day = new Forecast[4]; static { for (int i = 0; i< day.length; i++){ day[i] = new Forecast(); } } public static String GetSmsMsg(){ String msg = ""; msg += city + ","; msg += current_condition + ", " + current_temp+". "; msg += day[0].day_of_week+“, ” + day[0].condition + “, ” + 、 day[0].high + "/" + day[0].low; return msg; } }
  • 52. 11.3 程序开发11.3.4 后台服务 Forecast.java文件的代码如下 package edu.hrbeu.WeatherDemo.Weather; import android.graphics.Bitmap; public class Forecast { public String day_of_week; public String low; public String high; public String image_url; public Bitmap image; public String condition; }
  • 53. 11.3 程序开发11.3.4 后台服务 数据获取模块:天气数据是从Google提供的Web Service中获取的,数据的获取地址是http://www.google.com/ig/api?hl=en&weather=New%20York New%20York表示获取纽约(New York)的天气数据, %20表示一个空格 读者可以替换New%20York,并将新的地址输入Web浏览器,在浏览器中可以直接看到XML格式的天气数据 在资源目录中的/res/xml/api.xml文件,就是2009年9月22日获取的纽约天气数据 在程序资源中保留api.xml文件,主要是用来帮助读者分析XML数据格式,在程序运行期间并不访问该文件
  • 54. 11.3 程序开发11.3.4 后台服务 api.xml文件的内容如下
  • 55. 11.3 程序开发11.3.4 后台服务
  • 56. 11.3 程序开发11.3.4 后台服务 \
  • 57. 11.3 程序开发11.3.4 后台服务 标签内的数据是天气预报的城市和时间等基本信息,标签内的是当时的天气状况,4个标签是未来四天的天气情况 在api.xml文件中,还提供了能够反映天气情况的图标地址,例如第19行、第27行和第35行等
  • 58. 11.3 程序开发11.3.4 后台服务 WeatherAdapter类实现了利用URL获取位图的私有函数GetURLBitmap(),以及用来下载和解析XML数据的公有函数GetWeatherData() 后台服务在调用GetWeatherData()函数解析Google提供的天气数据时,会不断调用GetURLBitmap()函数,将XML数据中的天气图标根据图标地址下载到本地保存
  • 59. 11.3 程序开发11.3.4 后台服务 GetURLBitmap()函数的代码如下 private static Bitmap GetURLBitmap(String urlString){ URL url = null; Bitmap bitmap = null; try { url = new URL("http://www.google.com" + urlString); } catch (MalformedURLException e){ e.printStackTrace(); } try{ HttpURLConnection conn = (HttpURLConnection) url.openConnection(); conn.connect(); InputStream is = conn.getInputStream(); bitmap = BitmapFactory.decodeStream(is); is.close();
  • 60. 11.3 程序开发11.3.4 后台服务 第12行代码构造了支持HTTP功能的URLConnection 第14行代码返回字节流 第15行代码使用字节流产生位图 第16行代码关闭字节流 }catch (IOException e){ e.printStackTrace(); } return bitmap; }
  • 61. 11.3 程序开发11.3.4 后台服务 GetWeatherData()函数首先根据指定的URL地址,从网络获取字节流数据,然后调用轻量级XML解析器XmlPullParser对天气数据进行解析,并将解析结果保存在Weather类的公有静态属性中 GetWeatherData()函数的代码如下 public static void GetWeatherData() throws IOException, Throwable { String queryString = "http://www.google.com/ig/api?weather=" + Config.CityName; URL aURL = new URL(queryString.replace(" ", "%20")); URLConnection conn = aURL.openConnection(); conn.connect(); InputStream is = conn.getInputStream(); XmlPullParserFactory factory = XmlPullParserFactory.newInstance() factory.setNamespaceAware(true); XmlPullParser parser = factory.newPullParser(); parser.setInput(is,"UTF-8");
  • 62. 11.3 程序开发11.3.4 后台服务 12. int dayCounter = 0; while(parser.next()!= XmlPullParser.END_DOCUMENT){ String element = parser.getName(); if (element != null && element.equals("forecast_information")){ while(true){ int eventCode = parser.next(); element = parser.getName(); if (eventCode == XmlPullParser.START_TAG){ if (element.equals("city")){ Weather.city = parser.getAttributeValue(0); 23. }else if (element.equals("current_date_time")){ Weather.current_date_time = parser.getAttributeValue(0); 25 .} }
  • 63. 11.3 程序开发11.3.4 后台服务 28. if (element.equals("forecast_information") && eventCode == XmlPullParser.END_TAG){ break; } } }
  • 64. 11.3 程序开发11.3.4 后台服务 if (element != null && element.equals("current_conditions")){ while(true){ int eventCode = parser.next(); element = parser.getName(); if (eventCode == XmlPullParser.START_TAG){ if (element.equals("condition")){ Weather.current_condition = parser.getAttributeValue(0);\ }else if (element.equals("temp_f")){ Weather.current_temp = parser.getAttributeValue(0); }else if (element.equals("humidity")){ Weather.current_humidity = parser.getAttributeValue(0); }else if (element.equals("wind_condition")){ Weather.current_wind = parser.getAttributeValue(0); }else if (element.equals("icon")){ Weather.current_image_url = parser.getAttributeValue(0); Weather.current_image = GetURLBitmap(Weather.current_image_url); } }
  • 65. 11.3 程序开发11.3.4 后台服务 if (element.equals("current_conditions") && eventCode == XmlPullParser.END_TAG){ break; } } }
  • 66. 11.3 程序开发11.3.4 后台服务 if (element != null && element.equals("forecast_conditions")){ while(true){ int eventCode = parser.next(); element = parser.getName(); if (eventCode == XmlPullParser.START_TAG){ if (element.equals("day_of_week")){ Weather.day[dayCounter].day_of_week = parser.getAttributeValue(0); }else if (element.equals("low")){ Weather.day[dayCounter].low = parser.getAttributeValue(0); }else if (element.equals("high")){ Weather.day[dayCounter].high = parser.getAttributeValue(0); }else if (element.equals("icon")){ Weather.day[dayCounter].image_url = parser.getAttributeValue(0); Weather.day[dayCounter].image = GetURLBitmap(Weather.day[dayCounter].image_url); }else if (element.equals("condition")){ Weather.day[dayCounter].condition = parser.getAttributeValue(0); }
  • 67. 11.3 程序开发11.3.4 后台服务 } if (element.equals("forecast_conditions") && eventCode == XmlPullParser.END_TAG){ dayCounter++; break; } } } } is.close(); }
  • 68. 11.3 程序开发11.3.4 后台服务 最后,在AndroidManifest.xml文件中注册WeatherService,并声明连接互联网和发送SMS短信的两个用户许可
  • 69. 11.3 程序开发11.3.5 用户界面 在用户界面设计上,采用可以在多个分页上快速切换的Tab标签页 WeatherDemo示例的Tab标签页将每个标签页与一个Activity关联在一起,这样做的好处就是可以将不同标签页的代码放在不同的文件中,而且每个标签页都可以有独立的选项菜单
  • 70. 11.3 程序开发11.3.5 用户界面 WeatherDemo类是继承TabActivity的Tab标签页,共设置3个标签页 TAB1标签页的标题为“天气预报”,关联的Activity为WeatherActivity TAB2标签页的标题为“历史数据”,关联Activity为HistoryActivity TAB3标签页的标题为“系统设置”,关联Activity为SetupActivity
  • 71. 11.3 程序开发11.3.5 用户界面 WeatherDemo.java文件的完整代码如下 package edu.hrbeu.WeatherDemo; import android.app.TabActivity; import android.content.Intent; import android.os.Bundle; import android.widget.TabHost; public class WeatherDemo extends TabActivity { @Override 10. public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); 12. TabHost tabHost = getTabHost(); tabHost.addTab(tabHost.newTabSpec("TAB1") .setIndicator("天气预报", getResources().getDrawable(R.drawable.tab_weather)) .setContent(new Intent(this, WeatherActivity.class)));
  • 72. 11.3 程序开发11.3.5 用户界面 WeatherDemo.java中的代码只是用户界面的框架,设置了Tab标签页的图标、标题和所关联的Activity,标签页中的具体显示内容还要依赖于每个Activity所设置的界面布局17. tabHost.addTab(tabHost.newTabSpec("TAB2") .setIndicator("历史数据",getResources().getDrawable(R.drawable.tab_history)) .setContent(new Intent(this, HistoryActivity.class))); tabHost.addTab(tabHost.newTabSpec("TAB3") .setIndicator("系统设置",getResources().getDrawable(R.drawable.tab_setup)) .setContent(new Intent(this, SetupActivity.class))); } }
  • 73. 11.3 程序开发11.3.5 用户界面 界面布局包含 WeatherActivity HistoryActivity SetupActivity WeatherActivity: 主要用来显示天气信息
  • 74. 11.3 程序开发11.3.5 用户界面 WeatherActivity在启动时并不能够显示最新的天气信息,用户需要通过选项菜单的“启动服务”开启后台服务,然后点击“刷新”获取最新的天气状况 选项菜单还提供“停止服务”和“退出”选项 WeatherActivity使用的布局文件是tab_weather.xml,这是个较为繁琐的界面布局,多次的使用了垂直和水平的线性布局 WeatherActivity的界面布局和代码并不难以理解,因此这里不再给出WeatherActivity.java和tab_weather.xml具体代码
  • 75. 11.3 程序开发11.3.5 用户界面 HistoryActivity:主要用来显示SQLite数据库中的短信服务信息,显示的内容包括发送者的手机号码、时间和回复短信内容 为了能够以列表的形式显示多行数据,并定制每行数据的布局,使用了以往章节没有介绍过的ListActivity(Android.app.ListActivity)
  • 76. 11.3 程序开发11.3.5 用户界面 ListActivity可以不通过setContentView()设置布局,也不必重载onCreate()函数,而直接将显示列表加载到ListActivity,增加了使用的便利性 在WeatherDemo示例中,仍然使用setContentView()设置布局,这样做的好处是可以在界面中设置更为复杂的显示元素,例如在列表上方增加了提示信息“SQLite数据库中的短信服务信息” 下方的代码是HistoryActivity.java文件的onCreate()函数中的设置布局和加载适配器的关键代码 setContentView(R.layout.tab_history); setListAdapter(dataAdapter);
  • 77. 11.3 程序开发11.3.5 用户界面 tab_history.xml是HistoryActivity的布局文件,下面先分析一下tab_history.xml的内容 tab_history.xml文件的完整代码如下
  • 78. 11.3 程序开发11.3.5 用户界面 tab_history.xml在代码的第12行至第16行增加了ListView控件,并使用系统的ID值“@android:id/list”,ListView的数据列配器是通过setListAdapter(dataAdapter)设置的
  • 79. 11.3 程序开发11.3.5 用户界面 ListView使用的是自定义布局,布局保存在data_row.xml文件中,data_row.xml的完整代码如下
  • 80. 11.3 程序开发11.3.5 用户界面 \
  • 81. 11.3 程序开发11.3.5 用户界面 Android提供的数据适配器仅允许保存字符串数组或列表对象,如果希望使用自定义布局,则需要实现自定义的数据适配器,并继承Android提供的BaseAdapter(Android.widget.BaseAdapter)对象 自定义的数据适配器在SmsAdapter.java文件中,其完整代码如下 package edu.hrbeu.WeatherDemo.SMS; import android.content.Context; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; 7. import android.widget.BaseAdapter; 8. import android.widget.TextView;
  • 82. 11.3 程序开发11.3.5 用户界面 import edu.hrbeu.WeatherDemo.DB.DBAdapter; import edu.hrbeu.WeatherDemo.R; public class SmsAdapter extends BaseAdapter{ private LayoutInflater mInflater; private static DBAdapter dbAdapter ; private static SimpleSms[] smsList ; public SmsAdapter(Context context){ mInflater = LayoutInflater.from(context); dbAdapter = new DBAdapter(context); dbAdapter.open(); smsList = dbAdapter.GetAllSms(); }
  • 83. 11.3 程序开发11.3.5 用户界面 public static void RefreshData(){ smsList = dbAdapter.GetAllSms(); } @Override public int getCount(){ if (smsList == null) return 0; else return smsList.length; } @Override public Object getItem(int position) { if (smsList == null) return 0; else return smsList[position]; }
  • 84. 11.3 程序开发11.3.5 用户界面 @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder; if(convertView == null){ convertView = mInflater.inflate(R.layout.data_row, null); holder = new ViewHolder(); holder.textRow01 = (TextView) convertView.findViewById(R.id.data_row_01); holder.textRow02 = (TextView) convertView.findViewById(R.id.data_row_02); convertView.setTag(holder); }
  • 85. 11.3 程序开发11.3.5 用户界面 else{ holder = (ViewHolder) convertView.getTag(); } if (smsList != null){ String row01Msg ="("+position+")" +" 发送者:"+ smsList[position].Sender+","+smsList[position].ReceiveTime; holder.textRow01.setText(row01Msg); holder.textRow02.setText(smsList[position].ReturnResult); } return convertView; } private class ViewHolder{ TextView textRow01; TextView textRow02; } }
  • 86. 11.3 程序开发11.3.5 用户界面 继承BaseAdapter类,则首先要重载4个函数,包括getCount()、getItem()、getItemId()和getView() LayoutInflater是将XML文件中的布局映射为View对象的类,在代码第14行进行了声明,在代码第51行,将data_row.xml文件映射为View对象 代码第70行和第71行的内容,需要对应data_row.xml文件中的界面元素
  • 87. 11.3 程序开发11.3.5 用户界面 SetupActivity:主要用来保存和恢复用户设置的运行参数 第一次启动或恢复缺省设置(在选项菜单中)后,界面上会显示系统的缺省设置,包括城市名称、更新频率、是否提供短信服务、是否记录短信服务数据信息和短信服务的关键字
  • 88. 11.3 程序开发11.3.5 用户界面 SetupActivity.java文件中,主要功能集中在RestoreDefaultSetup()、UpdateUI()和SaveConfig()三个函数上 RestoreDefaultSetup()用来恢复系统的缺省配置 UpdateUI()会根据保存在Config类中的数据更新SetupActivity的界面控件 SaveConfig()根据界面配置更改Config类,然后调用数据库适配器的DBAdapter.SaveConfig()函数,将Config类中的配置数据写入数据库
  • 89. 11.3 程序开发11.3.5 用户界面 private void RestoreDefaultSetup(){ Config.LoadDefaultConfig(); UpdateUI(); dbAdapter.SaveConfig(); } private void UpdateUI(){ cityNameView.setText(Config.CityName); refreshSpeedView.setText(Config.RefreshSpeed); smsServiceView.setChecked(Config.ProvideSmsService.equals("true")?true:false); saveSmsInfoView.setChecked(Config.SaveSmsInfo.equals("true")?true:false); keyWorkView.setText(Config.KeyWord); }
  • 90. 11.3 程序开发11.3.5 用户界面 private void SaveConfig(){ Config.CityName = cityNameView.getText().toString().trim(); Config.RefreshSpeed = refreshSpeedView.getText().toString(); if (smsServiceView.isChecked()){ Config.ProvideSmsService = "true"; }else{ Config.ProvideSmsService = "false"; } if (saveSmsInfoView.isChecked()){ Config.SaveSmsInfo = "true"; } else{ Config.SaveSmsInfo = "false"; } Config.KeyWord = keyWorkView.getText().toString().trim(); dbAdapter.SaveConfig(); }
  • 91. 11.3 程序开发11.3.5 用户界面 为了使定义的Activity和ListActivity生效,在AndroidManifest.xml文件中注册所有定义的组件