这篇主要记录的是AIDL的基本使用。
主要参考:Android:学习AIDL,这一篇文章就够了(上)
Android中独有的进程间通信方式是Binder,前面学习Service的时候就知道,Service服务端返回给客户端的就是一个Binder,通过这个Binder完成服务端与客户端之间的通信。
这个服务端和客户端可以在同一个进程中,也可以在不同的进程中,对于相同的进程中的情况前面已经说了就是继承BInder类并将服务端的Service返回即可,但是对于不同的进程之间通信,这个就不行了,前面已经说了使用Messenger,不过只能传递消息,使用AIDL能够实现的就多了,可以进行方法调用。
首先,AIDL,Android Interface Definition Language,Android接口定义语言,简单的讲,就是用来定义接口的。
然后,只需要定义接口就可以了进行进程间通信了么?不能这么说,但是这个语言是Android定义好的,语法也比较简单,跟java很类似,我们只需要在其中简单的定义好我们需要的接口,并写好相应的序列化的类,然后通过IED就会帮我们生成好相应的文件,我们就可以使用这个来进行进程间通信。
使用AIDL进行进程间通信的主要流程:
如果只需要传递基本的数据类型,那么就没什么问题,直接写AIDL接口文件即可,但是如果想传递自定义的类对象,那么就需要考虑到序列化的问题,因为基本的数据类型默认都是可序列化的,但是我们自己写的类是不可以的,因此就需要实现Android中的Parcelable接口,至于这个接口的使用,这个在前面学习Intent传递对象的时候就已经说过了。
首先,需要创建一个实现了Parcelable接口的bean
然后写这个bean对应的AIDL文件:
// Book.aidl
package com.example.gsq.servicetest.bean;
parcelable Book;
然后写我们需要的接口AIDL文件:
// BookManager.aidl
package com.example.gsq.servicetest.bean;
import com.example.gsq.servicetest.bean.Book;
interface BookManager {
List<Book> getBookList();
void addBook(in Book book);
}
一定要注意包名的问题!
Book.aidl的包名需要与bean的包名一致
且BookManager.aidl中一定要导入Book.aidl的包(虽然它们已经在一个包下)
还有就是服务端与客户端的AIDL文件的包名也需要一致(不同的应用,所以建议放在单独的包中)
尤其是使用AS的,一定要注意。
在AS中写好了AIDL文件以及java bean之后,只需要点Rebuild Project,AS就会自动生成接口的java类:
至于具体的文件内容该怎么写,需要注意的点,开头的那篇文章里面介绍的很详细,建议去学习学习。
到这里就可以开始创建Service服务端了
首先创建一个BookManager.Stub的实例,在其中实现我们前面定义的接口,然后在onBind中将这个实例返回给客户端。
public class MyService3 extends Service {
//CopyOnWriteArrayList支持冰法读写,确保线程同步,其中的方法都是synchronized的
private CopyOnWriteArrayList<Book> bookList = new CopyOnWriteArrayList<>();
private final BookManager.Stub bookManager = new BookManager.Stub() {
@Override
public List<Book> getBookList() throws RemoteException {
if(bookList == null){
bookList = new CopyOnWriteArrayList<>();
}
return bookList;
}
@Override
public void addBook(Book book) throws RemoteException {
if(bookList == null){
bookList = new CopyOnWriteArrayList<>();
}
if(book != null && !bookList.contains(book)){
bookList.add(book);
}
}
};
@Override
public IBinder onBind(Intent intent) {
Log.i("test_out","----->onBind");
bookList.add(new Book("serverbook", "ser"));
return bookManager;
}
@Override
public boolean onUnbind(Intent intent) {
Log.i("test_out","----->onUnbind");
return true;
}
}
我们还可以在service创建的时候进行一些数据的初始化等。
然后是客户端,通过ServiceConnection获取到服务端传来的Binder,转换为BookManager,然后就可以进行服务端的方法调用了。
public void addBook(){
try {
bookManager.addBook(new Book("book1", "tom"));
bookManager.addBook(new Book("book2", "jack"));
for(Book b : bookManager.getBookList()){
Log.i("test_out","----->" + b.toString());
}
} catch (RemoteException e) {
e.printStackTrace();
}
}
ServiceConnection serviceConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
bookManager = BookManager.Stub.asInterface(service);
}
@Override
public void onServiceDisconnected(ComponentName name) {
Log.i("test_out","----->onServiceDisconnected");
}
};
程序运行的记过如下:
服务端:
客户端点击对了addBook的按钮之后:
只是写了一下AIDL的基本使用,由于刚开始学,理解不够深入,有不正确的地方请批评指正。