Skip to content

Latest commit

 

History

History
185 lines (168 loc) · 7.31 KB

服务.md

File metadata and controls

185 lines (168 loc) · 7.31 KB

服务

服务的两种开启方式:

  1. startService();开启服务.
    开启服务后 服务就会长期的后台运行,即使调用者退出了.服务仍然在后台继续运行.服务和调用者没有什么关系, 调用者是不可以访问服务里面的方法.
  2. bindService();绑定服务. 服务开启后,生命周期与调用者相关联.调用者挂了,服务也会跟着挂掉.不求同时生,但求同时死.调用者和服务绑定在一起,调用者可以间接的调用到服务里面的方法.

AIDL

本地服务:服务代码在本应用中
远程服务:服务在另外一个应用里面(另外一个进程里面) aidl: android interface defination language IPC implementation : inter process communication

服务混合调用的生命周期

开启服务后再去绑定服务然后再去停止服务,这时服务是无法停止了.必须先解绑服务然后再停止服务,在实际开发中会经常采用这种模式,开启服务(保证服务长期后台运行) --> 绑定服务(调用服务的方法) --> 解绑服务(服务继续在后台运行) --> 停止服务(服务停止),服务只会被开启一次,如果已经开启后再去执行开启操作是没有效果的。

绑定服务调用方法的原理

  1. 定义一个接口,里面定义一个方法
    public interface IService {
        public void callMethodInService();//通过该类中提供一个方法,让自定 
    义的类实现这个接口
    }
  1. 在服务中自定义一个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();
    	}
    }
  1. 服务的调用类中将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

上面介绍了绑定服务调用服务中方法的原理,对于远程服务的绑定也是这样, 但是这个远程服务是在另外一个程序中的,在另外一个程序中定 义的这个接口, 在另外一个程序中是拿不到的,就算是我们在自己的应用 中也定义一个一模一样 的接口,但是由于两个程序的报名不同,这两个接口也是不一样的,为了解决这个 问题,谷歌的工程师给提供了aidl,我们将定义的这个接口的.java改成 .aidl, 然后将这个接口中的权限修饰符去掉,在另一个程序中拷贝这个aidl文 件,然后放到同一个包名中,由于Android中通过包名来区分应用程序,这两个 aidl的包名一样,系统会认为两个程序中的接口是同一个,这样就能够在另一 个程序中将参数强转成这个接口,在使用aidl文件拷贝到自己的工程之后会自动 生成一个接口类,这个接口类中有 一个内部类Stub该类继承了Binder并实现了 这个接口,所以我们在自定义 IBinder的实现类时只需让自定义的类继承Stub类 即可.

  1. 远程服务中定义一个接口,改成aidl
    package com.seal.test.service;
    
    interface IService {
         void callMethodInService();
    }
  1. 远程服务中定义一个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("我是远程服务里面的方法");
    	}
	}
  1. 在其他程序中想要绑定这个服务并且调用这个服务中的方法的时候首先要拷贝 这个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
            }
        }
    }