本地服务:服务代码在本应用中
远程服务:服务在另外一个应用里面(另外一个进程里面)
aidl: android interface defination language
IPC implementation : inter process communication
开启服务后再去绑定服务然后再去停止服务,这时服务是无法停止了.必须先解绑服务然后再停止服务,在实际开发中会经常采用这种模式,开启服务(保证服务长期后台运行) –> 绑定服务(调用服务的方法) –> 解绑服务(服务继续在后台运行) –> 停止服务(服务停止),服务只会被开启一次,如果已经开启后再去执行开启操作是没有效果的。
1. 定义一个接口,里面定义一个方法
public interface IService { public void callMethodInService();//通过该类中提供一个方法,让自定义的类实现这个接口 }
2.在服务中自定义一个IBinder的实现类,让这个类继承Binder(Binder是IBinder的默认适配器),由于这个自定义类是私有的,为了其他类中能拿到该类,我们要定义一个接口,提供一个方法,让IBinder类去实现该接口,并在相应方法中调用自己要供别人调用的方法。
public class TestService extends Service { @Override public IBinder onBind(Intent intent) { System.out.println("onbind"); return new MyBinder(); } private class MyBinder extends Binder implements IService{ public void callMethodInService(){ //实现该方法,去调用服务中的方法 methodInService(); } } //服务中的方法 public void methodInService(){ Toast.makeText(this, "我是服务里面的春哥,巴拉布拉!", 0).show(); } }
3.服务的调用类中将onServiceConnected
方法中的第二个参数强转成接口
public class DemoActivity extends Activity { private Intent intent; private Myconn conn; private IService iService; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); intent = new Intent(this,TestService.class); setContentView(R.layout.main); } public void start(View view) { startService(intent); } public void stop(View view) { stopService(intent); } public void bind(View view) { conn = new Myconn(); //1.绑定服务 传递一个conn对象.这个conn就是一个回调接口 bindService(intent, conn, Context.BIND_AUTO_CREATE); } public void unbind(View view) { unbindService(conn); } //调用服务中的方法 public void call(View view){ iService.callMethodInService(); } private class Myconn implements ServiceConnection{ //当服务被成功绑定的时候调用的方法. @Override public void onServiceConnected(ComponentName name, IBinder service) {//第二个参数就是服务中的onBind方法的返回值 iService = (IService) service; } @Override public void onServiceDisconnected(ComponentName name) { } } }
上面介绍了绑定服务调用服务中方法的原理,对于远程服务的绑定也是这样,但是这个远程服务是在另外一个程序中的,在另外一个程序中定义的这个接口, 在另外一个程序中是拿不到的,就算是我们在自己的应用 中也定义一个一模一样 的接口,但是由于两个程序的报名不同,这两个接口也是不一样的,为了解决这个 问题,谷歌的工程师给提供了aidl,我们将定义的这个接口的.java
改成.aidl
, 然后将这个接口中的权限修饰符
都去掉,在另一个程序中拷贝这个aidl文件,然后放到同一个包名中,由于Android
中通过包名来区分应用程序,这两个aidl
的包名一样,系统会认为两个程序中的接口是同一个,这样就能够在另一个程序中将参数强转成这个接口,在使用aidl
文件拷贝到自己的工程之后会自动生成一个接口类,这个接口类中有 一个内部类Stub
该类继承了Binder
并实现了这个接口,所以我们在自定义 IBinder的实现类
时只需让自定义的类继承Stub类
即可.
1. 远程服务中定义一个接口,改成aidl
package com.seal.test.service; interface IService { void callMethodInService(); }
2. 远程服务中定义一个Ibinder的实现类
,让这个实现类继承上面接口的Stub类,并
在onBind
方法中返回这个自定义类对象
public class RemoteService extends Service { @Override public IBinder onBind(Intent intent) { System.out.println("远程服务被绑定"); return new MyBinder(); } private class MyBinder extends IService.Stub{ @Override public void callMethodInService() { methodInService(); } } public void methodInService(){ System.out.println("我是远程服务里面的方法"); } }
2. 在其他程序中想要绑定这个服务并且调用这个服务中的方法的时候首先要拷贝这个aidl
文件到自己的工程,然后再ServiceConnection
的实现类中将这个参数使用asInterface
方法转成接口,通过这样来得到接口,从而调用接口中的方法
public class CallRemoteActivity extends Activity { private Intent service; private IService iService; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); service = new Intent(); service.setAction("com.itheima.xxxx"); } public void bind(View veiw){ bindService(service, new MyConn(), BIND_AUTO_CREATE); } public void call(View view){ try { iService.callMethodInService(); } catch (RemoteException e) { e.printStackTrace(); } } private class MyConn implements ServiceConnection{ @Override public void onServiceConnected(ComponentName name, IBinder service) { iService = IService.Stub.asInterface(service); } @Override public void onServiceDisconnected(ComponentName name) { // TODO Auto-generated method stub } } }