zoukankan      html  css  js  c++  java
  • aidl

    通过startService()和stopService()启动关闭服务。适用于服务和访问者之间没有交互的情况。如果服务和访问者之间需要方法调用或者传递参数,侧需要使用bindService()和unbindService()方法启动关闭服务。

    采用Context.bindService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onBind()方法,这个时候访问者和服务绑定在一起。 如果访问者要与服务进行通信,那么,onBind()方法必须返回Ibinder对象。如果访问者退出了,系统就会先调用服务的onUnbind()方法,接着调用onDestroy()方法。如果调用bindService()方法前服务已经被绑定,多次调用bindService()方法并不会导致多次创建服务及绑定(也就是说onCreate()和onBind()方法并不会被多次调用)。如果访问者希望与正在绑定的服务解除绑定,可以调用unbindService()方法,调用该方法也会导致系统调用服务的onUnbind()-->onDestroy()方法。

    Activity与服务进行通信,开发人员通常把通信方法定义在接口里,然后让Ibinder对象实现该接口,而Activity通过该接口引用服务onBind()方法返回的Ibinder对象,然后调用Ibinder对象里自定义的通信方法。例子如下:

    本例是一个本地服务,即服务与Activity在同一个应用内部。

    接口:

    1 public interface ICountService 
    2 { 
    3     public int getCount(); 
    4 }

    服务类:

    public class CountService extends Service {
    private boolean quit;
    private int count;
    private ServiceBinder serviceBinder = new ServiceBinder();
    
    public class ServiceBinder extends Binder implements ICountService {
        @Override
        public int getCount() {
            return count;
        }
    }
    @Override
    public IBinder onBind(Intent intent) {
        return serviceBinder;
    }
    @Override
    public void onCreate() {
        super.onCreate();
        new Thread(new Runnable() {
            @Override
            public void run() {
                while (!quit) {
                    try {
                    Thread.sleep(1000);
                    } catch (InterruptedException e) {}
                    count++;
                }
            }
        }).start();
    }
    
    @Override
    public void onDestroy() {
        super.onDestroy();
        this.quit = true;
    }
    }

    客户端Activity:

    public class ClientActivity extends Activity {
        private ICountService countService;
    
        @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.main);
            this.bindService(new Intent(this, CountService.class), this.serviceConnection, BIND_AUTO_CREATE);
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            this.unbindService(serviceConnection);
        }    
        
        private ServiceConnection serviceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                countService = (ICountService) service;//对于本地服务,获取的实例和服务onBind()返回的实例是同一个
                int i = countService.getCount();            
                Log.v("CountService", "Count is " + i);
            }
            @Override
            public void onServiceDisconnected(ComponentName name) {
                countService = null;
            }
        };
    }

    在Android中, 每个应用程序都有自己的进程,当需要在不同的进程之间传递对象时,该如何实现呢? 显然, Java中是不支持跨进程内存共享的。因此要传递对象, 需要把对象解析成操作系统能够理解的数据格式, 以达到跨界对象访问的目的。在JavaEE中,采用RMI通过序列化传递对象。在Android中, 则采用AIDL(Android Interface Definition Language:接口定义语言)方式实现。

    AIDL是一种接口定义语言,用于约束两个进程间的通讯规则,供编译器生成代码,实现Android设备上的两个进程间通信(IPC)。AIDL的IPC机制和EJB所采用的CORBA很类似,进程之间的通信信息,首先会被转换成AIDL协议消息,然后发送给对方,对方收到AIDL协议消息后再转换成相应的对象。由于进程之间的通信信息需要双向转换,所以android采用代理类在背后实现了信息的双向转换,代理类由android编译器生成,对开发人员来说是透明的。

    实现进程通信,一般需要下面四个步骤:(请见页面下方备注栏)

    假设A应用需要与B应用进行通信,调用B应用中的download(String path)方法,B应用以Service方式向A应用提供服务。需要下面四个步骤:

    1> 在B应用中创建*.aidl文件,aidl文件的定义和接口的定义很相类,如:在cn.itcast.aidl包下创建IDownloadService.aidl文件,内容如下:

    package cn.itcast.aidl;

    interface IDownloadService

    {

        void download(String path);

    }

    当完成aidl文件创建后,eclipse会自动在项目的gen目录中同步生成IDownloadService.java接口文件。接口文件中生成一个Stub的抽象类,里面包括aidl定义的方法,还包括一些其它辅助方法。值得关注的是asInterface(IBinder iBinder),它返回接口类型的实例,对于远程服务调用,远程服务返回给客户端的对象为代理对象,客户端在onServiceConnected(ComponentName name, IBinder service)方法引用该对象时不能直接强转成接口类型的实例,而应该使用asInterface(IBinder iBinder)进行类型转换。

    编写Aidl文件时,需要注意下面几点:

    1.接口名和aidl文件名相同。

    2.接口和方法前不用加访问权限修饰符public,private,protected等,也不能用final,static。

    3.Aidl默认支持的类型包话java基本类型(int、long、boolean等)和(String、List、Map、CharSequence),使用这些类型时不需要import声明。对于List和Map中的元素类型必须是Aidl支持的类型。如果使用自定义类型作为参数或返回值,自定义类型必须实现Parcelable接口。

    4.自定义类型和AIDL生成的其它接口类型在aidl描述文件中,应该显式import,即便在该类和定义的包在同一个包中。

    5.在aidl文件中所有非Java基本类型参数必须加上in、out、inout标记,以指明参数是输入参数、输出参数还是输入输出参数。

    6.Java原始类型默认的标记为in,不能为其它标记。

    2> 在B应用中实现aidl文件生成的接口(本例是IDownloadService),但并非直接实现接口,而是通过继承接口的Stub来实现(Stub抽象类内部实现了aidl接口),并且实现接口方法的代码。内容如下:

    public class ServiceBinder extends IDownloadService.Stub
    {
        @Override
        public void download(String path) throws RemoteException
        {
            Log.i("DownloadService", path);
        }
    }

    3> 在B应用中创建一个Service(服务),在服务的onBind(Intent intent)方法中返回实现了aidl接口的对象(本例是ServiceBinder)。内容如下:

    public class DownloadService extends Service {
        private ServiceBinder serviceBinder = new ServiceBinder();
        @Override
        public IBinder onBind(Intent intent) {
            return serviceBinder;
        }
        public class ServiceBinder extends IDownloadService.Stub {
            @Override
            public void download(String path) throws RemoteException {
                Log.i("DownloadService", path);
            }        
        }
    }

    其他应用可以通过隐式意图访问服务,意图的动作可以自定义,AndroidManifest.xml配置代码如下:

    <service android:name=".DownloadService" >
        <intent-filter>
            <action android:name="cn.itcast.process.aidl.DownloadService" />
        </intent-filter>
    </service>

    4> 把B应用中aidl文件所在package连同aidl文件一起拷贝到客户端A应用,eclipse会自动在A应用的gen目录中为aidl文件同步生成IDownloadService.java接口文件,接下来就可以在A应用中实现与B应用通信,代码如下:

    public class ClientActivity extends Activity {
        private IDownloadService downloadService;
    
        @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.main);
            this.bindService(new Intent("cn.itcast.process.aidl.DownloadService"), this.serviceConnection, BIND_AUTO_CREATE);//绑定到服务
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            this.unbindService(serviceConnection);//解除服务
        }    
        
        private ServiceConnection serviceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                downloadService = IDownloadService.Stub.asInterface(service);
                try {
                    downloadService.download("http://www.itcast.cn");
                } catch (RemoteException e) {
                    Log.e("ClientActivity", e.toString());
                }
            }
            @Override
            public void onServiceDisconnected(ComponentName name) {
                downloadService = null;
            }
        };
    }

    aidl默认支持的类型包话java基本类型(int、long、boolean等)和(String、List、Map、CharSequence),如果要传递自定义的类型该如何实现呢?

    要传递自定义类型,首先要让自定义类型支持parcelable协议,实现步骤如下:

    1>自定义类型必须实现Parcelable接口,并且实现Parcelable接口的public void writeToParcel(Parcel dest, int flags)方法 。

    2>自定义类型中必须含有一个名称为CREATOR的静态成员,该成员对象要求实现Parcelable.Creator接口及其方法。

    3> 创建一个aidl文件声明你的自定义类型。 Parcelable接口的作用:实现了Parcelable接口的实例可以将自身的状态信息(状态信息通常指的是各成员变量的值)写入Parcel,也可以从Parcel中恢复其状态。 Parcel用来完成数据的序列化传递。

    进程间传递自定义类型的实现过程请参见页面下方备注栏:

    1> 创建自定义类型,并实现Parcelable接口,使其支持parcelable协议。如:在cn.itcast.domain包下创建Person.java:

    package cn.itcast.domain;
    import android.os.Parcel;
    import android.os.Parcelable;
    public class Person implements Parcelable
        private Integer id;
        private String name;
        
        public Person(){}
        public Person(Integer id, String name) {
            this.id = id;
            this.name = name;
        }
        public Integer getId() {
            return id;
        }
        public void setId(Integer id) {
            this.id = id;
        }
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }    
        @Override
        public int describeContents() {
            return 0;
        }
        @Override
        public void writeToParcel(Parcel dest, int flags) {//把javanbean中的数据写到Parcel
            dest.writeInt(this.id);
            dest.writeString(this.name);
        }
    //添加一个静态成员,名为CREATOR,该对象实现了Parcelable.Creator接口
        public static final Parcelable.Creator<Person> CREATOR = new Parcelable.Creator<Person>(){
            @Override
            public Person createFromParcel(Parcel source) {//从Parcel中读取数据,返回person对象
                return new Person(source.readInt(), source.readString());
            }
            @Override
            public Person[] newArray(int size) {
                return new Person[size];
            }
        };
    }

    2> 在自定义类型所在包下创建一个aidl文件对自定义类型进行声明,文件的名称与自定义类型同名。

    package cn.itcast.domain;

    parcelable Person;

    3> 在接口aidl文件中使用自定义类型,需要使用import显式导入,本例在cn.itcast.aidl包下创建IPersonService.aidl文件,内容如下:

    package cn.itcast.aidl;
    import cn.itcast.domain.Person;
    interface IPersonService {
          void save(in Person person);
    }

    4> 在实现aidl文件生成的接口(本例是IPersonService),但并非直接实现接口,而是通过继承接口的Stub来实现(Stub抽象类内部实现了aidl接口),并且实现接口方法的代码。内容如下:

    public class ServiceBinder extends IPersonService.Stub {
           @Override
           public void save(Person person) throws RemoteException {
        Log.i("PersonService", person.getId()+"="+ person.getName());
           }        
    }

    5> 创建一个Service(服务),在服务的onBind(Intent intent)方法中返回实现了aidl接口的对象(本例是ServiceBinder)。内容如下:

    public class PersonService extends Service {
        private ServiceBinder serviceBinder = new ServiceBinder();
        @Override
        public IBinder onBind(Intent intent) {
            return serviceBinder;
        }
    public class ServiceBinder extends IPersonService.Stub {
           @Override
           public void save(Person person) throws RemoteException {
        Log.i("PersonService", person.getId()+"="+ person.getName());
           }
    }
    }

    其他应用可以通过隐式意图访问服务,意图的动作可以自定义,AndroidManifest.xml配置代码如下:

    <service android:name=".PersonService" >
        <intent-filter>
            <action android:name="cn.itcast.process.aidl.PersonService " />
        </intent-filter>
    </service>

    6> 把应用中的aidl文件和所在package一起拷贝到客户端应用的src目录下,eclipse会自动在客户端应用的gen目录中为aidl文件同步生成IPersonService.java接口文件,接下来再把自定义类型文件和类型声明aidl文件及所在package一起拷贝到客户端应用的src目录下。

    最后就可以在客户端应用中实现与远程服务的通信,代码如下:

    public class ClientActivity extends Activity {
        private IPersonService personService;
    
        @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.main);
            this.bindService(new Intent("cn.itcast.process.aidl.PersonService"), this.serviceConnection, BIND_AUTO_CREATE);//绑定到服务
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            this.unbindService(serviceConnection);//解除服务
        }    
        
        private ServiceConnection serviceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                personService = IPersonService.Stub.asInterface(service);
                try {
                    personService.save(new Person(56,"liming"));
                } catch (RemoteException e) {
                    Log.e("ClientActivity", e.toString());
                }
            }
            @Override
            public void onServiceDisconnected(ComponentName name) {
                personService = null;
            }
        };
    }

    转自:http://wenku.baidu.com/view/7c996d14cc7931b765ce15e0.html

  • 相关阅读:
    如何消除一个数组里面重复的元素?
    行内元素有哪些?块级元素有哪些? 空(void)元素有那些?
    简述一下src与href的区别
    请说出三种减少页面加载时间的方法
    SQL大全
    11.Hibernate 拦截器
    10.Hibernate 批处理
    9.Hibernate 缓存
    8.Hibernate 原生 SQL
    7.Hibernate 标准查询
  • 原文地址:https://www.cnblogs.com/jimwind/p/2956035.html
Copyright © 2011-2022 走看看