Android编程-IntentService使用广播与Activity通信
2015-11-30 20:12
453 查看
Android编程过程中,经常涉及到后台程序,一个长时间运行的后台程序使用Android提供的Service将是一个很好的选择,然而在众多Service中,IntentService最为常用,也最为简单,但是在使用IntentService时,将服务产生的结果反馈给Activity却不是一件容易的事情。一般情况下,Activity与Service通信有两种方法,其一是通过绑定Binder对象,其二是通过broadcast(广播)的形式。本文将介绍一种属于第二种方案的特例,该方法使用IntentService作为后台Service,使用动态注册broadcast接收器作为通信桥梁,这样将可以省去很大一部分代码量,并且将通信变得更简单。
1.继承IntentService实现onHandleIntent方法
继承IntentService至少需要做两件事,第一,实现一个没有参数的默认构造器,并且在构造函数内调用含有String参数的父类(IntentService)构造器;第二,也是最重要的地方,就是实现OnHandleIntent(Intent
intent)方法,该方法是IntentService的一个虚函数,当后台服务启动后,它将在后台服务的新线程调用,而不是主线程。我们实现该方法完成我们需要在后台完成的任务,如下载任务,其中intent参数将在启动服务时由我们创建并传递进来,这样在OnHandleIntent方法中我们就可以根据Intent的参数内容做出不同的后台响应。
启动IntentService可以使用Activity的startService方法:
2.实现广播和动态注册广播
广播接收器动态注册与XML注册的最大区别是作用范围的大小,XML定义方法可以在整个系统范围内进行广播并接收,而动态注册仅当在本应用可以使用。因此在特定的Activity与Service之间通信,使用动态注册广播接收更为方便。
动态注册广播接收器的过程,首先需要实现一个广播接收器,如MsgReceiver,重写OnReceive方法,该方法的intent参数是后台Service发送广播时提供的反馈内容。接下来,注册接收器可以使用Activity的registerReceiver方法,该方法接收一个BroadcastReceiver对象和一个IntentFilter对象,这里我们使用MsgReceiver的实例作为BroadcastReceiver对象参数,而IntentFilter
对象是用来标记(关联)接收器的,以便在发送广播后系统的正确找到接收器。这里我们实例一个IntentFilter对象,并调用addAction方法传递一个SERVICE_RECEIVER常量,表示我们接受具有SERVICE_RECEIVER常量表示的操作(Action)的广播,最后我们需要在onDestroy方法中停止服务和注销接收器。
当后台服务的(下载)任务完成时,我们就可以发送相应的广播来通知Activity,然后广播接收器就可以在OnReceive方法对接收到的Intent参数作出相应的响应,如UI更新。如下:
这里我们创建一个具有SERVICE_RECEIVER常量表示的操作(Action)的Intent对象,然后使用Context.SendBroadcast方法发送该Intent对象代表的广播。
到现在为止,使用上述方法已经完全可以做到Activity与Service通信了,但是有一个问题,这里通信采用的是Intent进行数据内容传输,如果我们是下载任务的话,很有可以需要传递一个自定义的类,那么怎么进行传输呢?这就需要使用到Parcelable接口,实现该接口可以将你的自定义类序列化,然后放入Intent中进行传输。
3.实现Parcelable接口
假设我们下载任务完成后得到的是如下Item类的一组对象,那么我们将要怎么样将其传递给广播接收器呢?
首先,必须要实现Item的序列化,序列化是通过实现Parcelable接口而实现的,这里我们将上面的类改造为ItemParcelable类,并实现Parcelable接口:
首先重写writeToParcel方法,将你的对象(ItemParcelable)序列化为一个Parcel对象,即:将类的数据写入外部提供的Parcel中,打包需要传递的数据到Parcel容器保存,以便从Parcel容器获取数据。接下来,重写describeContents方法,内容接口描述,默认返回0就可以。最后实例化静态内部对象CREATOR实现Parcelable.Creator
publicstatic final Parcelable.Creator CREATOR接口。注意,其中public static final一个都不能少,内部对象CREATOR的名称也不能改变,必须全部大写。需重写本接口中的两个方法:createFromParcel(Parcelin)实现从Parcel容器中读取传递数据值,封装成Parcelable对象返回逻辑层,newArray(int
size) 创建一个类型为T,长度为size的数组,仅一句话即可(return
new T[size]),供外部类反序列化本类数组使用,这里的T就是ItemParcelable。还需要注意的是writeToParcel和createFromParcel方法中序列化的顺序必须相同。
4.最后通过ItemParcelable传递数据
发送数据
接收数据
到现在为止,我们已经成功的完成Activity与Service的通信任务,最后还需要弄清楚一个概念,IntentService在第一次调用startService时创建服务,如果在IntentService还没有完成后台任务(onHandleIntent函数)时再次调用startService函数,那么不再创建服务,而是在任务队列添加一个任务(也可以理解为消息,其实就是将Intent包含的信息添加到一个队列中等待调用OnHandleIntent调用),等待上次任务完成后再继续完成任务队列的下一条任务,当任务队列所有任务执行完毕后,则销毁服务,所以Service并不是自始至终都在后台运行,而只在有任务输入的时候才运行,这也保证了设备内存的充分利用。
1.继承IntentService实现onHandleIntent方法
public class DownloadService extends IntentService { public static final String SERVICE_KEY_WORDS = "com.art.zok.DownloadService"; public DownloadService() { super("***"); } @Override protected void onHandleIntent(Intent intent) { String url = intent.getStringExtra("***"); downMethods(url); } }
继承IntentService至少需要做两件事,第一,实现一个没有参数的默认构造器,并且在构造函数内调用含有String参数的父类(IntentService)构造器;第二,也是最重要的地方,就是实现OnHandleIntent(Intent
intent)方法,该方法是IntentService的一个虚函数,当后台服务启动后,它将在后台服务的新线程调用,而不是主线程。我们实现该方法完成我们需要在后台完成的任务,如下载任务,其中intent参数将在启动服务时由我们创建并传递进来,这样在OnHandleIntent方法中我们就可以根据Intent的参数内容做出不同的后台响应。
启动IntentService可以使用Activity的startService方法:
public class MainClass extends Activity { @Override public void onCreate(Bundle savedInstanceState) { Intent i = new Intent(this, DownloadService.class); /*i.putExtra(****)此处可以添加相应的参数*/ startService(i); } }
2.实现广播和动态注册广播
广播接收器动态注册与XML注册的最大区别是作用范围的大小,XML定义方法可以在整个系统范围内进行广播并接收,而动态注册仅当在本应用可以使用。因此在特定的Activity与Service之间通信,使用动态注册广播接收更为方便。
public class MainClass extends Activity { public String SERVICE_RECEIVER = "com.art.zok.receiver"; private Intent intent; private MsgReceiver msgReceiver; public void onCreate(Bundle savedInstanceState) { // 动态注册广播接收器 msgReceiver = new MsgReceiver(); IntentFilter intentFilter = new IntentFilter(); intentFilter.addAction(SERVICE_RECEIVER); registerReceiver(msgReceiver, intentFilter); //启动后台服务 intent = new Intent(this, DownloadService.class); /*intent.putExtra(****)此处可以添加相应的参数*/ startService(intent); } @Override public void onDestroy() { // 停止服务 stopService(intent); // 注销广播 unregisterReceiver(msgReceiver); super.onDestroy(); } public class MsgReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { /*处理接收到的广播内容*/ } } }
动态注册广播接收器的过程,首先需要实现一个广播接收器,如MsgReceiver,重写OnReceive方法,该方法的intent参数是后台Service发送广播时提供的反馈内容。接下来,注册接收器可以使用Activity的registerReceiver方法,该方法接收一个BroadcastReceiver对象和一个IntentFilter对象,这里我们使用MsgReceiver的实例作为BroadcastReceiver对象参数,而IntentFilter
对象是用来标记(关联)接收器的,以便在发送广播后系统的正确找到接收器。这里我们实例一个IntentFilter对象,并调用addAction方法传递一个SERVICE_RECEIVER常量,表示我们接受具有SERVICE_RECEIVER常量表示的操作(Action)的广播,最后我们需要在onDestroy方法中停止服务和注销接收器。
当后台服务的(下载)任务完成时,我们就可以发送相应的广播来通知Activity,然后广播接收器就可以在OnReceive方法对接收到的Intent参数作出相应的响应,如UI更新。如下:
public class DownloadService extends IntentService { *** @Override protected void onHandleIntent(Intent intent) { *** downMethods(url); *** // 发送广播通知Activity Intent sendIntent = new Intent(MainClass.SERVICE_RECEIVER); *** getApplicationContext().sendBroadcast(sendIntent); } }
这里我们创建一个具有SERVICE_RECEIVER常量表示的操作(Action)的Intent对象,然后使用Context.SendBroadcast方法发送该Intent对象代表的广播。
到现在为止,使用上述方法已经完全可以做到Activity与Service通信了,但是有一个问题,这里通信采用的是Intent进行数据内容传输,如果我们是下载任务的话,很有可以需要传递一个自定义的类,那么怎么进行传输呢?这就需要使用到Parcelable接口,实现该接口可以将你的自定义类序列化,然后放入Intent中进行传输。
3.实现Parcelable接口
假设我们下载任务完成后得到的是如下Item类的一组对象,那么我们将要怎么样将其传递给广播接收器呢?
public class Item { long id; String title; String img; int gallertClass; int count; int rcount; int fcount; long time; int size; ***** }
首先,必须要实现Item的序列化,序列化是通过实现Parcelable接口而实现的,这里我们将上面的类改造为ItemParcelable类,并实现Parcelable接口:
public class ItemParcelable implements Parcelable { long id; String title; String img; int gallertClass; int count; int rcount; int fcount; long time; int size; public ItemParcelable() {} public long getId() { return id; } public long setId(long id) { this.id = id; } public String getTitle() { return title; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeLong(id); dest.writeString(title); dest.writeString(img); dest.writeInt(gallertClass); dest.writeInt(count); dest.writeInt(rcount); dest.writeInt(fcount); dest.writeLong(time); dest.writeInt(size); } /** *实现Parcelable接口的类中, *必须有一个实现了Parcelable.Creator *接口的静态常量成员字段,并且它的名 *字必须为CREATOR **/ public static final Parcelable.Creator<ItemParcelable> CREATOR = new Parcelable.Creator<ItemParcelable>() { @Override public ItemParcelable createFromParcel(Parcel in) { // 从包裹中读出数据 ItemParcelable item = new ItemParcelable(); item.setId(in.readLong()); item.setTitle(in.readString()); item.setImg(in.readString()); item.setGallertClass(in.readInt()); item.setCount(in.readInt()); item.setRcount(in.readInt()); item.setFcount(in.readInt()); item.setTime(in.readLong()); item.setSize(in.readInt()); return item; } @Override public ItemParcelable[] newArray(int size) { return new ItemParcelable[size]; } }; }
首先重写writeToParcel方法,将你的对象(ItemParcelable)序列化为一个Parcel对象,即:将类的数据写入外部提供的Parcel中,打包需要传递的数据到Parcel容器保存,以便从Parcel容器获取数据。接下来,重写describeContents方法,内容接口描述,默认返回0就可以。最后实例化静态内部对象CREATOR实现Parcelable.Creator
publicstatic final Parcelable.Creator CREATOR接口。注意,其中public static final一个都不能少,内部对象CREATOR的名称也不能改变,必须全部大写。需重写本接口中的两个方法:createFromParcel(Parcelin)实现从Parcel容器中读取传递数据值,封装成Parcelable对象返回逻辑层,newArray(int
size) 创建一个类型为T,长度为size的数组,仅一句话即可(return
new T[size]),供外部类反序列化本类数组使用,这里的T就是ItemParcelable。还需要注意的是writeToParcel和createFromParcel方法中序列化的顺序必须相同。
4.最后通过ItemParcelable传递数据
发送数据
@Override protected void onHandleIntent(Intent intent) { *** ArrayList<ItemParcelable> items = downMethods(***); // 发送广播通知Activity Intent sendIntent = new Intent(MainClass.SERVICE_RECEIVER); sendIntent.putParcelableArrayListExtra(SERVICE_KEY_WORDS, items); getApplicationContext().sendBroadcast(sendIntent); }
接收数据
@Override public void onReceive(Context context, Intent intent) { ArrayList<ItemParcelable> items = intent .getParcelableArrayListExtra(DownloadService.SERVICE_KEY_WORDS); /** *for(ItemParcelable item : items) { * item.getId(); * *** *} **/ **** }
到现在为止,我们已经成功的完成Activity与Service的通信任务,最后还需要弄清楚一个概念,IntentService在第一次调用startService时创建服务,如果在IntentService还没有完成后台任务(onHandleIntent函数)时再次调用startService函数,那么不再创建服务,而是在任务队列添加一个任务(也可以理解为消息,其实就是将Intent包含的信息添加到一个队列中等待调用OnHandleIntent调用),等待上次任务完成后再继续完成任务队列的下一条任务,当任务队列所有任务执行完毕后,则销毁服务,所以Service并不是自始至终都在后台运行,而只在有任务输入的时候才运行,这也保证了设备内存的充分利用。
相关文章推荐
- Android学习笔记-XML解析和JSON
- Android4.4 RIL的AT命令增加流程
- Android屏幕适配
- Android 获取手机GPS
- Android开发总结笔记 Menu(菜单) 1-1-18
- android开发之Notification_通知栏消息
- Android Button的3D触摸效果
- Android 解决BitmapFactory.decodeFile(file) 报OOM问题
- 321android浏览器
- Android之异步处理机制
- ListView适配器BaseAdapter之getView的原理--convertView源码分析Android5.1
- Android 中 ToolBar箭头颜色设置
- 列表视图(ListView)——BaseAdapter
- Android短信----发送流程---框架层(Frameworks)
- android:service
- Android 扫描指定路径里的所有文件
- Android sqlite3工具的使用
- Android 图文混排(EditText)
- android 沉浸模式记录
- 通过Sql语句和android封装好的方法控制SQLite数据库增删改查