android菜鸟学习笔记28----Android中的Service生命周期及本地和远程服务绑定的实现
Service是Android中长期在后台运行的没有界面的组件,使用服务的优势在于:能够提高进程的优先级,系统不容易回收掉进程,即便回收了,内存充足的时候,会把进程重新创建。
1.服务的简单使用示例:
1.1.定义一个服务:
定义一个服务的方式是定义一个类继承自Service:
public class MyService extends Service { @Override public IBinder onBind(Intent intent) { // TODO Auto-generated method stub return null; } }
注意到onBind()方法是个抽象方法,因而必须重写。
1.2.注册服务:
四大组件之一的Service,在使用之前也必须在清单文件中注册。
<service android:name="cn.csc.service.MyService"></service>
1.3.在MainActivity中启动这个服务:
启动服务的方式很简单,可以如同启动Activity一样,通过Intent对象进行服务的启动,启动服务的方法是startService(Intent intent),下面的代码在MainActivity的onCreate()方法中启动服务:
protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = new Intent(this, MyService.class); startService(intent); }
1.4.运行效果:
2.观察服务的生命周期:
2.1修改MyService及MainActivity代码,重写生命周期回调的各个方法,输出标识信息:
MainActivity.java:
public class MainActivity extends ActionBarActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Log.i("MainActivity","onCreate()"); Intent intent = new Intent(this, MyService.class); startService(intent); } @Override protected void onStop() { // TODO Auto-generated method stub super.onStop(); Log.i("MainActivity","onStop()"); } @Override protected void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); Log.i("MainActivity","onDestroy()"); } @Override protected void onPause() { // TODO Auto-generated method stub super.onPause(); Log.i("MainActivity","onPause()"); } @Override protected void onResume() { // TODO Auto-generated method stub super.onResume(); Log.i("MainActivity","onResume()"); } @Override protected void onStart() { // TODO Auto-generated method stub super.onStart(); Log.i("MainActivity","onStart()"); } }
MyService.java:
public class MyService extends Service { @Override public IBinder onBind(Intent intent) { // TODO Auto-generated method stub Log.i("MyService","onBind()"); return null; } @Override public void onCreate() { // TODO Auto-generated method stub super.onCreate(); Log.i("MyService","onCreate()"); } @Override public int onStartCommand(Intent intent, int flags, int startId) { // TODO Auto-generated method stub Log.i("MyService","onStartCommand()"); return super.onStartCommand(intent, flags, startId); } @Override public void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); Log.i("MyService","onDestroy()"); } @Override public boolean onUnbind(Intent intent) { // TODO Auto-generated method stub Log.i("MyService","onUnbind()"); return super.onUnbind(intent); } }
2.2部署安装应用:
运行结果:
2.3退出当前应用:
查看运行中的进程:
服务仍然在运行。
点击Stop之后:
服务才停止运行。
总结:
采用startService()方式开启服务后,先后调用了服务的onCrate()和onStartCommand()方法,Activity被销毁之后,服务仍然继续运行,只有当手动停止掉服务时,才会调用onDestory()方法。
可见,start方式开启服务时, 一旦服务开启跟调用者(开启者)就没有任何关系了。
开启者退出后,服务仍然在后台长期的运行,开启者与服务没有任何关联,因而不能调用服务里面的方法。
2.4在MainActivity中调用bindService()绑定服务:
protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Log.i("MainActivity","onCreate()"); Intent intent = new Intent(this, MyService.class); bindService(intent, new MyConn(), BIND_AUTO_CREATE); }
查看运行结果:
退出应用:
总结:
采用绑定的方式开启服务,会先后调用:onCreate()、onBind(),Activity销毁后,Service紧跟着调用onunbind()、onDestory()也被销毁。
这种方式的服务跟随应用的结束而结束,不会长期驻留在后台,但是,这种方式下,Activity与Service产生了关联,因而可以调用Service提供的方法。那么在Activity中如何调用Service中的方法呢?
3.Activity调用本地服务方法:
主要步骤:
1)在本地服务中添加需要被Activity调用的方法
2)定义一个接口,约定Activity通过该接口的实例调用本地服务的方法,该接口的实例由本地服务的onBind()方法传递给Activity
3) 在服务的内部创建一个内部类实现约定的接口,并在onBind()方法中返回该内部类的实例,由于onBind()方法返回的是IBinder类型的实例,所以该内部类还需要继承一个Ibinder接口的实现类,一般继承Binder类
4)在activity 绑定服务:bindService();
5)在服务成功绑定的时候 会执行一个方法 onServiceConnected 传递过来一个 IBinder对象
5)将传回的IBinder对象强制类型转化成约定接口的实例,从而调用接口里面的方法。
具体编码步骤:
1)修改MyService代码,添加一个供Activity调用的方法:
public void serviceMethod(){ Toast.makeText(getApplicationContext(), "本地服务方法被调用", Toast.LENGTH_SHORT).show(); }
2)定义一个双方约定的接口,用于调用前面添加的serviceMethod()方法:
public interface IMiddlePerson { public void callServiceMethod(); }
3)在MyService中定义一个内部类继承自Binder并实现上面的约定接口:
private class MiddlePerson extends Binder implements IMiddlePerson{ @Override public void callServiceMethod() { // TODO Auto-generated method stub serviceMethod(); } }
4)修改MyService的onBind()方法,返回上面定义的内部类的实例:
public IBinder onBind(Intent intent) { // TODO Auto-generated method stub Log.i("MyService","onBind()"); return new MiddlePerson(); }
5)在MainActivity中添加一个存放约定接口的引用的字段:
private IMiddlePerson imp;
6)修改MainActivity的布局文件,添加一个按钮:
<Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="call" android:layout_gravity="center_horizontal" android:text="@string/call_native_service" />
在MainActivity中添加public void call(View view)方法,用于按钮的点击回调:
public void call(View view){ imp.callServiceMethod(); }
7)修改MyConn的onServiceConnected方法,将IBinder类型的参数赋值给imp:
public void onServiceConnected(ComponentName name, IBinder service) { // TODO Auto-generated method stub Log.i("MainActivity","服务绑定了"); imp = (IMiddlePerson) service; }
8)查看运行结果:
注: 如果又想服务能长期在后台运行,又想能够调用服务中的方法,则应如下顺序处理:
1)start方式开启服务(保证服务长期后台运行)
2)bind方式绑定服务(保证调用服务的方法)
在需要停止服务时,则需要按如下顺序:
3)unbind解除绑定服务
4)stopService停止服务。
这里,就不在测试了。
4.Activity调用远程服务的方法:
这里简单区分下本地服务和远程服务:
远程服务:调用者和服务在不同的工程代码里面。
本地服务:调用者和服务在同一个工程代码里面。
如果像调用同一个工程中的本地服务方法一样,绑定一个远程服务,直接会报错。这是因为绑定远程服务的方法与绑定本地服务的方法有所不同:
Android定义了aidl:android interface definition language 安卓接口定义语言来实现远程服务的绑定。
与绑定本地服务的主要不同在于:
约定的接口的定义不能带有任何访问修饰符,而且后缀名要改为.aidl,之后会自动生成一个名为Stub的内部类,服务中定义的内部类继承该内部类即可。
具体编码步骤:
4.1定义能远程绑定的服务:
1)定义IMiddlePeson.aidl:
package cn.csc.service; interface IMiddlePerson { void callServiceMethod(); }
将在gen目录下自动生成:
打开自动生成的IMiddlePerson.java:
可以看到自动生成了Stub内部抽象类
2)在服务中添加要被远程调用的方法:
public void serviceMethod(){ Log.i("MyService","远程服务方法被调用了。。。"); }
3)在服务中定义内部类继承IMiddlePerson.Stub:
private class MiddlePerson extends IMiddlePerson.Stub{ @Override public void callServiceMethod() { // TODO Auto-generated method stub serviceMethod(); } }
4)由于要被远程调用,所以不能使用显示意图了,需要在清单文件中注册服务时添加<intent-filter>生命能响应的动作:
<service android:name="cn.csc.service.MyService"> <intent-filter > <action android:name="cn.csc.service.MyService"/> </intent-filter> </service>
至此,一个可以被远程绑定的服务就完成了,先把它部署到模拟器上。
4.2在另一个项目中绑定该远程服务:
1)将约定的接口IMiddlePerson.aidl文件连同所在包一同复制到该项目的src目录中:
可以看到,同样自动生成了IMiddlePerson.java文件
2)定义存放接口引用的字段:
private IMiddlePerson imp;
3)使用隐式意图绑定远程服务:
Intent intent = new Intent(); intent.setAction("cn.csc.service.MyService"); bindService(intent, new MyConn(), BIND_AUTO_CREATE); 4)修改MyConn()中的onServiceConnected()方法: public void onServiceConnected(ComponentName name, IBinder service) { // TODO Auto-generated method stub Log.i("MainActivity","服务绑定了"); imp = IMiddlePerson.Stub.asInterface(service); }
调用远程服务的应用端也搞定了,然后部署,查看运行结果:
以上,就是今天学到的关于Service的一点知识。
android菜鸟学习笔记28----Android中的Service生命周期及本地和远程服务绑定的实现的更多相关文章
- android菜鸟学习笔记30----Android使用百度地图API(一)准备工作及在应用中显示地图
1.准备工作: 百度地图API是免费开放的,但是需要申请API Key: 1)先注册一个百度开发者帐号 2)进入百度开放服务平台http://developer.baidu.com/ 3)进入LBS云 ...
- android菜鸟学习笔记29----Android应用向用户发送提示信息的方式总结
常见的向用户发送提示信息的方式有3种,分别为: 1)发送Toast信息 2)弹出对话框 3)发送通知 总结如下: 方式1:发送Toast信息: 这种方式最简单,在之前的学习中多次使用过.Toast是在 ...
- android菜鸟学习笔记27----Fragment的简单使用
1.Fragment的生命周期: 简单在新建一个MyFragment继承自Fragment,重写各个生命周期回调方法,各个方法中直接输出标识相关函数被调用的信息. 重写MainActivity的各个生 ...
- android菜鸟学习笔记24----与服务器端交互(一)使用HttpURLConnection和HttpClient请求服务端数据
主要是基于HTTP协议与服务端进行交互. 涉及到的类和接口有:URL.HttpURLConnection.HttpClient等 URL: 使用一个String类型的url构造一个URL对象,如: U ...
- android菜鸟学习笔记21----ContentProvider(一)ContentProvider的简单使用
ContentProvider是Android四大组件之一,它用来封装数据,并通过ContentResolver接口将数据提供给其他应用.只有当需要在多个应用之间共享数据时才会用到ContentPro ...
- android菜鸟学习笔记8----Activity(一)
Activity是android应用程序中重要的组件之一,常听到的android四大组件是Activity.Service.BroadcastReceiver和ContentProvider.它间接继 ...
- android菜鸟学习笔记7----android布局(二)
3.FrameLayout:帧布局 如同Flash或者photoshop中图层的概念,在上面的图层遮盖下面的图层,没被遮到的地方仍然显示出来. 右击res/layout,然后在弹出的菜单中选择new, ...
- android菜鸟学习笔记2----关于adb
adb : android debug bridge android调试桥 路径:adt-bundle目录/sdk/platform-tools/adb.exe 常见的adb命令: adb devic ...
- android菜鸟学习笔记31----Android使用百度地图API(二)获取地理位置及地图控制器的简单使用
1.获取当前地理位置: Android中提供了一个LocationManager的类,用于管理地理位置.不能通过构造函数获取该类的实例,而是通过Context的getSystemService(): ...
随机推荐
- mui 选项卡
方法一:通过css 实现选项卡 <div id="slider" class="mui-slider"> <div id="slid ...
- mysql的innodb数据库引擎详解
http://www.jb51.net/softjc/158474.html 这篇文章主要介绍了mysql的innodb数据库引擎,需要的朋友可以参考下 一.mysql体系结构和存储引擎 1. ...
- Yii2.0 下的 load() 方法的使用
一 问题 最近在使用 Yii2.0,遇到一个 bug:在 /models/OrderDetail.php add() 方法中调用 load() 方法加载数据,却加载不了. public functio ...
- RabbitMQ 学习笔记(一)特点
RabbitMQ 的具体特点 可靠性: RabbitMQ 使用一些机制来保证可靠性, 如持久化.传输确认及发布确认等. 令灵活的路由: 在消息进入队列之前,通过交换器来路由消息.对于典型的路由功能,R ...
- Apache Rewrite 规则详解
在开篇之前: 我想说这篇文章其实是我刚刚接触Rewrite的时候学习的文档,应属转载,但是在这里我不想写明原地址,原因是文章中大多数给出的配置命令经实验都是错误的.需要原文的可以在谷歌上搜索一下&qu ...
- DevStore开发人员服务有奖征文:小谈新浪微博开放平台
DevStore开发人员服务有奖征文:小谈新浪微博开放平台 笔者接入新浪微博开发平台也有一段时间了,对整个平台的接入也算比較熟悉,新浪提供了统一的API接口,能够让开发人员更方便的使用API来实现自己 ...
- Atitit避免出现空指针异常解决方案
Atitit避免出现空指针异常解决方案 1. Null的问题1 2. 强制区分一般引用vs 可空引用 vs 强制引用,或者说非空引用2 3. ?运算符(问号运算符) !感叹号运算符避免出现空指针异常, ...
- [ci]安装配置jenkins及其插件
后面尝试ms模式部署多台jenkins 安装jenkins:(hudson是jenkins的商业版) cd /etc/yum.repos.d/ wget http://pkg.jenkins.io/r ...
- MSP430WARE++的使用3:modbus模块的调用方法
MSP430WARE++的使用3:modbus模块的调用方法 MSP430WARE是一套基于C++语言的开源的MSP430层次化软件架构,支持多种外设. 本文将介绍modbus模块驱动程序的调 ...
- FreeMarker中在list中加入if判断
例如list中遍历releaseitem,在ri中获取audit的值,如果audit的值为0则表示正在审核中,如果为1则表示审核通过,如果为2则表示未审核. <#list releaseitem ...