天天看点

什么是 AIDL?aidl 服务器端和客户端项目结构详解

作者:愿天堂没有代码

什么是 AIDL?

Android 系统中的进程之间不能共享内存,因此,需要提供一些机制在不同进程之间进行数据通信。

为了使其他的应用程序也可以访问本应用程序提供的服务,Android 系统采用了远程过程调用(Remote Procedure Call,RPC)方式来实现。与很多其他的基于RPC的解决方案一样,Android使用一种接口定义语言(Interface Definition Language,IDL)来公开服务的接口。我们知道4个Android应用程序组件中的3个(Activity、BroadcastReceiver和ContentProvider)都可以进行跨进程访问,另外一个Android应用程序组件Service同样可以。因此,将这种可以跨进程访问的服务称为AIDL(Android Interface Definition Language)服务

AIDL 服务的使用

  AIDL 和其他的 IDL 类似,它允许你定义程序接口,以便客户端与服务器端通过 IPC 机制交互。在 android 上面,一个进程一般不能访问另外进程的内存。因此,Android 平台将这些跨进程访问的对象分解成操作系统能够识别的简单对象。并为跨应用访问而特殊编排和整理这些对象。用于编排和整理这些对象的代码编写起来十分冗长,所以 Android 的 AIDL 提供了相关工具来自动生成这些代码。今天,我们开始AIDL的学习。

定义 AIDL 接口

为了测试,我写了 aidl 服务器端和 aidl 客户端!

aidl 服务器端

项目结构如下:

什么是 AIDL?aidl 服务器端和客户端项目结构详解

一、 创建.aidl文件:IRemoteAIDL.aidl

// IRemoteAIDL.aidl
package com.huhx.linux;

// Declare any non-default types here with import statements

interface IRemoteAIDL {
    int add(int a, int b);
}           

二、 定义服务类:RemoteService.java

package com.huhx.linux.aidlserver;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;

import com.huhx.linux.IRemoteAIDL;

public class RemoteService extends Service {
    private final static String TAG = "RemoteService";

    private IRemoteAIDL.Stub binder = new IRemoteAIDL.Stub() {
        @Override
        public int add(int a, int b) throws RemoteException {
            Log.i(TAG, "a = " + a + ", b = " + b);
            return a + b;
        }
    };

    @Override
    public IBinder onBind(Intent intent) {
        return binder;
    }
}           

三、 在AndroidManifest.xml文件中声明服务:

<service android:name=".RemoteService" android:enabled="true" android:exported="true" />           

aidl客户端

项目结构如下:

什么是 AIDL?aidl 服务器端和客户端项目结构详解

一、 创建与服务器端一样的aidl文件: IRemoteAIDL.aidl

// IRemoteAIDL.aidl
package com.huhx.linux;

// Declare any non-default types here with import statements

interface IRemoteAIDL {
    int add(int a, int b);
}           

二、 在MainActivity中,得到远程服务并调用它的方法。

package com.huhx.linux.aidlclient;

import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;

import com.huhx.linux.IRemoteAIDL;

public class MainActivity extends AppCompatActivity {
    private IRemoteAIDL iRemoteAIDL;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        binServices();
    }

    private ServiceConnection conn = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iRemoteAIDL = IRemoteAIDL.Stub.asInterface(service);
        }
        @Override
        public void onServiceDisconnected(ComponentName name) {
            iRemoteAIDL = null;
        }
    };

    // 绑定服务
    private void binServices() {
        Intent intent = new Intent();
        intent.setComponent(new ComponentName("com.huhx.linux.aidlserver", "com.huhx.linux.aidlserver.RemoteService"));
        bindService(intent, conn, Context.BIND_AUTO_CREATE);
    }

    // 处理相加的方法
    public void addMethod(View view) {
        try {
            int result = iRemoteAIDL.add(3, 4);
            Toast.makeText(MainActivity.this, "3 + 4 = " + result, Toast.LENGTH_SHORT).show();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
}           

AIDL的简单使用之通过 IPC 传递对象

如果要跨进程传递某个类,可以通过 IPC 接口来实现。 不过,请务必确保在 IPC 通道的对端可以识别该类的代码,该类必须支持 Parcelable 接口。支持 Parcelable 接口非常重要,因为这使得 Android 系统可将对象分解为能够跨进程组装的原生数据。

可按以下步骤创建支持 Parcelable 协议的类:

  1. 必须实现Parcelable 接口。
  2. 实现 writeToParcel 方法,参数为当前对象的状态,并写入一个 Parcel中。
  3. 在类中添加一个名为 CREATOR 的静态成员变量,即为一个实现了 Parcelable.Creator 接口的对象。
  4. 最后,创建 .aidl 文件,声明该 Parcelable 类(如下述 Rect.aidl 文件所示)。

如果采用自定义编译方式,请不要把 .aidl 文件加入编译项目。与 C 语言的头文件类似, .aidl 文件不会被编译。AIDL 利用上述方法和成员变量来分解和组装对象。

我们在上述的项目基础上做修改,服务器端:

什么是 AIDL?aidl 服务器端和客户端项目结构详解

一、 修改的IRemoteAIDL.aidl文件:

// IRemoteAIDL.aidl
package com.huhx.linux;

import com.huhx.linux.Person;

interface IRemoteAIDL {
    int add(int a, int b);

    List<Person> addPerson(in Person person);
}           

二、 增加Person.aidl文件:

// Person.aidl
package com.huhx.linux;

parcelable Person;           

三、 增加Person.java文件,注意它的位置是com.huhx.linux;

package com.huhx.linux;

import android.os.Parcel;
import android.os.Parcelable;

/**
 * Created by Linux on 2016/5/9.
 */
public class Person implements Parcelable {
    public String username;
    public String password;

    public Person(String username, String password) {
        this.username = username;
        this.password = password;
    }
    protected Person(Parcel in) {
        username = in.readString();
        password = in.readString();
    }
    public static final Creator<Person> CREATOR = new Creator<Person>() {
        @Override
        public Person createFromParcel(Parcel in) {
            return new Person(in);
        }
        @Override
        public Person[] newArray(int size) {
            return new Person[size];
        }
    };
    @Override
    public int describeContents() {
        return 0;
    }
    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(username);
        dest.writeString(password);
    }
}           

这里,之前是放在com.huhx.linux.aidlclient下的,但是处理起来总是报错。不知道是否要求aidl的包名与序列化类的包名一致?

四、 修改RemoteService.java文件

// 添加一个方法
public void addPerson(View view) {
    Person person1 = new Person("Linux", "123456");
    Person person2 = new Person("Huhx", "654321");
    int length = 0;
    try {
        iRemoteAIDL.addPerson(person1);
        length = iRemoteAIDL.addPerson(person2).size();

        Toast.makeText(MainActivity.this, "list size: " + length, Toast.LENGTH_SHORT).show();
    } catch (RemoteException e) {
        e.printStackTrace();
    }
}           

结语

有需要文中完整代码或者更多 Android 相关学习资料的小伙伴可以;可以私信发送"Android 进阶",即可获取一份Android 系列性技术学习手册;希望这份手册能够给大家学习 Android 带来一些帮助

什么是 AIDL?aidl 服务器端和客户端项目结构详解

好了,以上就是今天要分享的内容,大家觉得有用的话,可以点赞分享一下;如果文章中有什么问题欢迎大家指正;欢迎在评论区或后台讨论哈~

继续阅读