AIDL的简单使用和注意事项

概述

AIDL(Android interface definition Language)——Android 接口定义语言, 是 Android 提供的一种进程间通信 (IPC) 机制。可以利用它定义客户端与服务端使用进程间通信 (IPC) 进行相互通信时都认可的编程接口。 在 Android 上,一个进程通常无法访问另一个进程的内存。 尽管如此,进程需要将其对象分解成操作系统能够识别的原语,并将对象编组成跨越边界的对象。 编写执行这一编组操作的代码是一项繁琐的工作,因此 Android 会使用 AIDL 来处理。如果你有阅读源码的习惯,你会发现在Android系统提供的服务中大量使用了AIDL机制,使得其他进程能够访问并使用系统提供的服务。

使用步骤

服务端通过一个前台的Service结合SqliteDatabase数据持久化技术实现了一个简单的学生管理系统,提供学生的增删改查功能。客户端通过AIDL技术绑定服务端,进而访问和使用服务端提供的服务,对学生进行增删改查操作。同时,对服务端数据的改变进行监听,以便客户端做出相应的响应。下面分别从服务端和客户端的实现,一步一步讲解对应的实现。

服务端的实现

(1)创建服务端.aidl文件

  • 自定义数据封装类

派生Parcelable自定义了Student类用于封装学生相关信息,里面包含了预先设定的几个属性。

public class Student implements Parcelable {
    private long id;
    private String name;
    private int gender;
    private int age;
    private int score;
    ...
    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeLong(this.id);
        dest.writeString(this.name);
        dest.writeInt(this.gender);
        dest.writeInt(this.age);
        dest.writeInt(this.score);
    }

    protected Student(Parcel in) {
        this.id = in.readLong();
        this.name = in.readString();
        this.gender = in.readInt();
        this.age = in.readInt();
        this.score = in.readInt();
    }

    public static final Creator<Student> CREATOR = new Creator<Student>() {
        @Override
        public Student createFromParcel(Parcel source) {
            return new Student(source);
        }

        @Override
        public Student[] newArray(int size) {
            return new Student[size];
        }
    };
}

如果在AIDL文件中用到了自定义的Parcelable对象,必须创建一个和它同名的AIDL文件,并在其中声明它为Parcelable。

// Student.aidl
package com.android.peter.aidlservicedemo.bean;
// 注意:文件中Student声明处parcelable为小写 
parcelable Student;
  • 创建数据变化监听接口文件
// IOnDataChangeListener.aidl
package com.android.peter.aidlservicedemo;

interface IOnDataChangeListener {
    void onDataChange();
}
  • 创建服务接口文件

除了基本数据类型,其他类型的参数都需要标上方向类型:in(输入), out(输出), inout(输入输出)。

// IStudentManager.aidl
package com.android.peter.aidlservicedemo;
import com.android.peter.aidlservicedemo.bean.Student;
import com.android.peter.aidlservicedemo.IOnDataChangeListener;

interface IStudentManager {
    List<Student> getStudentList();
    long addStudent(in ContentValues contentValues);
    int deletedStudent(String whereClause, in String[] whereArgs);
    int updateStudent(in ContentValues contentValues, String whereClause, in String[] whereArgs);
    List<Student> queryStudent(in String[] columns, String selection,in String[] selectionArgs,
        String groupBy, String having,String orderBy, String limit);
    void registerDataChangeListener(IOnDataChangeListener listener);
    void unregisterDataChangeListener(IOnDataChangeListener listener);
}

(2)实现服务端接口

public class StudentManagerService extends Service {
    ...
    private RemoteCallbackList<IOnDataChangeListener> mListenerList = new RemoteCallbackList<>();
    private IBinder mService = new IStudentManager.Stub() {
        // 检查调用进程的权限
        @Override
        public boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException {
            int checkPermission = checkCallingOrSelfPermission(PERMISSION_STUDENT_MANAGER_SERVICE);
            if( checkPermission != PackageManager.PERMISSION_GRANTED) {
                Log.i(TAG,"Do not have permission !");
                return false;
            }

            String packageName = null;
            String[] packages = getPackageManager().getPackagesForUid(getCallingUid());
            if(packages != null && packages.length > 0) {
                packageName = packages[0];
            }

            if (packageName != null && !packageName.startsWith("com.android.peter")) {
                Log.i(TAG,"Package name must be contains \"com.android.peter\" !");
                return false;
            }

            return super.onTransact(code, data, reply, flags);
        }

        @Override
        public List<Student> getStudentList() throws RemoteException {
            Log.d(TAG,"getStudentList");
            return StudentListDaoImpl.getInstance(mContext).getAllStudent();
        }

        @Override
        public long addStudent(ContentValues contentValues) throws RemoteException {
            Log.d(TAG,"addStudent contentValues = " + contentValues);
            long id = StudentListDaoImpl.getInstance(mContext).insert(contentValues);
            // the row ID of the newly inserted row, or -1 if an error occurred
            if(id != -1) {
                notifyDataChanged();
            }

            return id;
        }

        @Override
        public int deletedStudent(String whereClause, String[] whereArgs) throws RemoteException {
            Log.d(TAG,"deletedStudent whereClause = " + whereClause + " , whereArgs = " + whereArgs);
            int num = StudentListDaoImpl.getInstance(mContext).delete(whereClause,whereArgs);
            // the number of rows affected if a whereClause is passed in, 0 otherwise.
            if(num > 0) {
                notifyDataChanged();
            }

            return num;
        }

        @Override
        public int updateStudent(ContentValues contentValues, String whereClause, String[] whereArgs) throws RemoteException {
            Log.d(TAG,"deletedStudent contentValues = " + contentValues + " , whereClause = " + whereClause + " , whereArgs = " + whereArgs);
            int num = StudentListDaoImpl.getInstance(mContext).update(contentValues,whereClause,whereArgs);
            // the number of rows affected
            if(num > 0) {
                notifyDataChanged();
            }

            return num;
        }

        @Override
        public List<Student> queryStudent(String[] columns, String selection,
                                   String[] selectionArgs, String groupBy, String having,
                                   String orderBy, String limit) throws RemoteException {
            Log.d(TAG,"queryStudent columns = " + columns + " , selection = " + selection + " , selectionArgs = " + selectionArgs
                    + " , groupBy = " + groupBy + " , having = " + having + " , orderBy = " + orderBy + " , limit = " + limit);
            return StudentListDaoImpl.getInstance(mContext).query(columns,selection,selectionArgs,groupBy,having,orderBy,limit);
        }

        @Override
        public void registerDataChangeListener(IOnDataChangeListener listener) throws RemoteException {
            mListenerList.register(listener);
        }

        @Override
        public void unregisterDataChangeListener(IOnDataChangeListener listener) throws RemoteException {
            mListenerList.unregister(listener);
        }
    };
    ...
    // 数据发生变化时候调用
    private void notifyDataChanged() {
        mListenerList.beginBroadcast();
        int N = mListenerList.getRegisteredCallbackCount();
        for(int i = 0 ; i < N ; i++) {
            try {
                mListenerList.getBroadcastItem(i).onDataChange();
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        mListenerList.finishBroadcast();
    }
   ...
}

(3)公开相关接口文件

客户端的实现

(1)把相关接口文件拷贝到相应位置
需要把所有相关的文件和类全部拷贝到客户端对应的位置,切记不要修改完整路径和文件名,否则会编译不过。

(2)绑定服务、获得服务

public class ClientActivity extends AppCompatActivity {
    private final static String TAG = "peter.ClientActivity";

    private final static String REMOTE_SERVICE_PACKAGE_NAME = "com.android.peter.aidlservicedemo";
    private final static String REMOTE_SERVICE_CLASS_NAME = "com.android.peter.aidlservicedemo.StudentManagerService";

    private final static int MESSAGE_DATA_CHANGED = 20180804;

    @SuppressLint("HandlerLeak")
    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            switch (msg.what) {
                case MESSAGE_DATA_CHANGED:
                    Log.i(TAG,"I have received a data changed message!");
                    break;
                default:
                    // do nothing
            }
        }
    };
    private IStudentManager mRemoteService;
    private IOnDataChangeListener mOnDataChangeLister = new IOnDataChangeListener.Stub() {
        @Override
        public void onDataChange() throws RemoteException {
            Log.i(TAG,"onDataChange");
            // running in Binder's thread pool,could be used to do long-time task,
            // and could not to access to UI thread directly
            mHandler.sendEmptyMessage(MESSAGE_DATA_CHANGED);
        }
    };
    private IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {
        @Override
        public void binderDied() {
            Log.i(TAG,"Remote service is died !");
            if(mRemoteService == null) {
                return;
            }

            mRemoteService.asBinder().unlinkToDeath(mDeathRecipient,0);
            mRemoteService = null;

            // rebind remote service
            bindRemoteService();
        }
    };
    private ServiceConnection mSC = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.i(TAG,"onServiceConnected name = " + name);
            // 获得服务端服务
            mRemoteService = IStudentManager.Stub.asInterface(service);
            if(mRemoteService != null) {
                try {
                    // 设置死亡代理
                    mRemoteService.asBinder().linkToDeath(mDeathRecipient,0);
                    // 注册监听器
                    mRemoteService.registerDataChangeListener(mOnDataChangeLister);
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            } else {
                Log.i(TAG,"Connect error!");
            }

            // onServiceConnected is running in main thread,
            // can not do long-time task
            new Thread(new Runnable() {
                @Override
                public void run() {
                    insertStudent();
                    queryStudent();
                    updateStudent();
                    deleteStudent();
                }
            }).start();
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.i(TAG,"onServiceDisconnected name = " + name);
            mRemoteService = null;
        }
    };

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

        bindRemoteService();
    }

    // bind remote service
    private void bindRemoteService() {
        Intent intent = new Intent();
        intent.setComponent(new ComponentName(REMOTE_SERVICE_PACKAGE_NAME, REMOTE_SERVICE_CLASS_NAME));
        bindService(intent,mSC, Context.BIND_AUTO_CREATE);
    }

    // inset three students
    private void insertStudent() {
        Log.i(TAG,"insertStudent");
        ContentValues peter = new ContentValues();
        peter.put(StudentTable.COLUMN_NAME,"peter");
        peter.put(StudentTable.COLUMN_GENDER,0);
        peter.put(StudentTable.COLUMN_AGE,33);
        peter.put(StudentTable.COLUMN_SCORE,100);

        ContentValues lemon = new ContentValues();
        lemon.put(StudentTable.COLUMN_NAME,"lemon");
        lemon.put(StudentTable.COLUMN_GENDER,1);
        lemon.put(StudentTable.COLUMN_AGE,30);
        lemon.put(StudentTable.COLUMN_SCORE,100);

        ContentValues baoyamei = new ContentValues();
        baoyamei.put(StudentTable.COLUMN_NAME,"baoyamei");
        baoyamei.put(StudentTable.COLUMN_GENDER,1);
        baoyamei.put(StudentTable.COLUMN_AGE,30);
        baoyamei.put(StudentTable.COLUMN_SCORE,90);

        try {
            if(mRemoteService != null) {
                mRemoteService.addStudent(peter);
                mRemoteService.addStudent(lemon);
                mRemoteService.addStudent(baoyamei);

                List<Student> studentList = mRemoteService.getStudentList();
                Log.i(TAG,"studentList = " + studentList.toString());
            }
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

    // query the student who's age is 30
    private void queryStudent() {
        Log.i(TAG,"queryStudent");
        if(mRemoteService != null) {
            try{
                List<Student> queryList = mRemoteService.queryStudent(StudentTable.TABLE_COLUMNS,StudentTable.COLUMN_AGE + "=?",
                        new String[]{"30"},null,null,null,null);
                Log.i(TAG,"queryList = " + queryList.toString());
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
    }

    // update the student who's score is 90
    private void updateStudent() {
        Log.i(TAG,"updateStudent");
        if(mRemoteService != null) {
            ContentValues lemon = new ContentValues();
            lemon.put(StudentTable.COLUMN_SCORE,100);
            try {
                mRemoteService.updateStudent(lemon,StudentTable.COLUMN_NAME + "=?",new String[]{"baoyamei"});
                List<Student> studentList = mRemoteService.getStudentList();
                Log.i(TAG,"studentList = " + studentList.toString());
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
    }

    // delete the student who's name is baoyamei
    private void deleteStudent() {
        Log.i(TAG,"deleteStudent");
        if(mRemoteService != null) {
            try{
                mRemoteService.deletedStudent(StudentTable.COLUMN_SCORE + "=?",new String[]{"100"});
                List<Student> studentList = mRemoteService.getStudentList();
                Log.i(TAG,"studentList = " + studentList.toString());
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if(mRemoteService != null && mRemoteService.asBinder().isBinderAlive()) {
            try {
                mRemoteService.unregisterDataChangeListener(mOnDataChangeLister);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        if(mSC != null) {
            unbindService(mSC);
        }
    }
}

小结

本文结合实例扼要的介绍了AIDL的使用方法,关于SqliteDatabase的使用可以参考我的另一篇文章——Android数据存储之SQLiteDatabase。使用过程中应该注意的地方:

  • 在定义Parcelable类和对应的.aidl文件的名字以及完整路径必须保持一致,否则编译会报错。同时,注意在.aidl文件中的parcelable声明为小写。
  • 除了基本数据类型,其他类型的参数都需要标上方向类型:in(输入), out(输出), inout(输入输出)。
  • 需要把所有相关的文件和类全部拷贝到客户端对应的位置,切记不要修改完整路径和文件名,否则会编译不过。
  • 在实现监听器的时候,推荐选用RemoteCallbackList类作为服务端存储监听器列表的容器。进程间是不允许直接传递对象的,是通过序列化和反序列化来实现的,这也是为什么在定义自定义数据类型时需要派生Parcelable类。RemoteCallbackList内部是以CallBack的IBinder对象作为key值,以CallBack对象作为value值进行存储的,服务端反序列化后生成的CallBack对象其实是新生成的,但是底层对应IBinder并没有改变,客户端和服务端的CallBack对象是通过IBinder来关联的。
  • 类似于访问网络,与服务端交互的时候很有可能也是耗时操作,最好不要在UI线程使用,有可能导致ANR。
  • 通过单实例保证服务端全局只有一个SQLiteOpenHelper对象与数据库连接,解决数据库多线程并发访问问题。
  • 从安全角度出发,需要为你的服务添加权限控制和包名检测,只有具有权限和指定包名的应用才能访问服务。添加权限控制的方法有多种,比如可以在AndroidManifest中添加权限声明,然后在Service声明处添加android:permission属性。或是在实现服务端接口的时候重写onTransact方法,这个方法在每次服务被调用的时候都会先被调用,可以在里面判断权限和包名。

更多的细节可以参看《Android开发艺术探索》关于AIDL的章节,里面对技术细节讲解的非常详细。

AIDLClientDemo
AIDLServiceDemo

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,719评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,337评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,887评论 0 324
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,488评论 1 266
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,313评论 4 357
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,284评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,672评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,346评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,644评论 1 293
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,700评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,457评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,316评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,706评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,261评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,648评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,859评论 2 335

推荐阅读更多精彩内容

  • Jianwei's blog 首页 分类 关于 归档 标签 巧用Android多进程,微信,微博等主流App都在用...
    justCode_阅读 5,880评论 1 23
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,121评论 25 707
  • 一、IPC简介 (1)IPC是Inter-Process Communication的缩写,含义为进程间通信或者跨...
    遥遥的远方阅读 7,169评论 0 3
  • 最近写的两篇日志是应付差事的,最近写的几条说说是情不自禁的,唯有今晚的数语将是心真实意的。最近两天生命状态差到了极...
    一只学术小狐狸阅读 246评论 0 2
  • 脑子迟钝,估计是为自己的自律性差找借口,很多道理自己知道,也知道自己的缺点,却没有去把事情做好,也没有改...
    居有定所的流浪人阅读 147评论 0 0