Android-服务(AIDL通讯)

本文学习目标

  1. 学会使用AIDL与远程服务进行通讯

AIDL介绍

AIDL是Android中IPC(Inter-Process Communication)跨进程通信的一种,AIDL是Android Interface definition Language的缩写。
通过AIDL,可以让本地调用远程服务的接口就像调用本地接口那么简单,让用户无需关注内部细节,只需要实现自己的业务逻辑接口,内部复杂的参数序列化发送,接收,客户端调用服务的逻辑,你都不需要关心。

因此,需要跨进程,我们就建立两个App,一个做服务端(提供接口),一个做为客户端。

步骤

服务端:

  1. 新建IMyAidlInterface.aidl文件
    通过Anroid Studio可以轻松新建aidl文件,File -> New -> AIDL
    此时,会多出一个aidl目录,以及我们程序的包名目录,还有aidl文件


    image.png
  2. 在AIDL文件里面添加接口
    basicTypes这个方法,是新建的时候有的,里面说明aidl默认支持的数据类型,如果是我们自定义的类型,需要一些特殊处理。

// IMyAidlInterface.aidl
package longma.com.myaidlserver;
import longma.com.myaidlserver.User;
// Declare any non-default types here with import statements

interface IMyAidlInterface {
    /**
     * Demonstrates some basic types that you can use as parameters
     * and return values in AIDL.
     */
    void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
            double aDouble, String aString);

    /**
    * 新增方法
    */
     String getName();
}

然后【Build Project】

  1. 新建服务
    新建服务,实现IMyAidlInterface.Stub接口,并且在onBind返回实现IMyAidlInterface.Stub接口的对象。
public class MyService extends Service {
    public final static String TAG = MyService.class.getSimpleName();
    public MyService() {
    }

    @Override
    public void onCreate() {
        Log.d(TAG, "onCreate");
        super.onCreate();
    }

    @Override
    public IBinder onBind(Intent intent) {
        Log.d(TAG, "onBind");
        return new MyBinder();
    }

    /**
     * 定义MyBinder实现IMyAidlInterface.Stub接口
     */
    class MyBinder extends IMyAidlInterface.Stub{

        @Override
        public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException {

        }

        @Override
        public String getName() throws RemoteException {
            return "MyService";
        }
    }
}

此时,服务端已经完成,运行,Run到模拟器或者设备上。等待客户端绑定调用。

客户端

  1. 把服务端的IMyAidlInterface.aidl文件拷贝到客户端(直接把aidl目录拷贝到客户端对应的目录)


    image.png
  2. 新建ServiceConnection,绑定服务
    IMyAidlInterface.Stub.asInterface(iBinder)获取IMyAidlInterface对象

Intent intent = new Intent();
intent.setComponent(new ComponentName("longma.com.myaidlserver", "longma.com.myaidlserver.MyService"));
bindService(intent, cnn, BIND_AUTO_CREATE);

IMyAidlInterface iMyAidlInterface;
ServiceConnection cnn = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
            iMyAidlInterface = IMyAidlInterface.Stub.asInterface(iBinder);
        }

        @Override
        public void onServiceDisconnected(ComponentName componentName) {

        }
    };
  1. 接口调用
        if (iMyAidlInterface != null){
            try {
                Toast.makeText(this, iMyAidlInterface.getName(), Toast.LENGTH_SHORT).show();
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }

客户端Run到模拟器或设备上,
效果:
点击绑定,点击调用


image.png

自定义类型

首先,我们自定义类型需要实现Parcelable接口,这样才可以跨进程序列化传输。

  1. 新建User类,通过Android Studio自动生成Parcelable接口代码。


    image.png
  2. 新建User.aidl
    去掉里面的接口定义,增加引用自定义的类。(Android Studio添加User.aidl会提示名字需要唯一,可能因为我们刚才建了User.java,你可以先命名为User1.aidl在重命名回User.aidl)

parcelable User;
image.png
  1. IMyAidlInterface.aidl添加接口
interface IMyAidlInterface {
    /**
     * Demonstrates some basic types that you can use as parameters
     * and return values in AIDL.
     */
    void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
            double aDouble, String aString);

    /**
    * 新增方法
    */
     String getName();

    /**
    *  新增方法
    */
     User getUser();
}

【Build Project】,会提示MyBinder有接口没有实现。

  1. 新增接口实现
class MyBinder extends IMyAidlInterface.Stub{
        @Override
        public String getName() throws RemoteException {
            return "MyService";
        }

        @Override
        public User getUser() throws RemoteException {
            User u = new User();
            u.setName("xzm");
            u.setAge(123);
            return u;
        }
    }

客户端:

  1. 更新所有aidl文件,
  2. 添加自定义数据类型User.java


    image.png

    User.java要放到java目录下,且包名要和服务端的一致。

【Build Project】,
我们就可以通过iMyAidlInterface调用getUser()方法了。

iMyAidlInterface.getUser()

推荐阅读更多精彩内容