Android 艺术开发探索笔记--进程间通信AIDL

1. Messenger是以串行方式处理客户端发来的消息,如果大量的消息发送到服务端,服务端仍然只能一个一个处理,如果有大量的并发请求,那么用Messenger就不太适合了。同时,Messenger的作用主要是为了传递消息,很多时候我们可能需要跨进程调用服务端的方法,这种时候我们可以使用AIDL来实现跨进程的方法调用。AIDL也是Messenger的底层实现,因此Messenger本质上也是AIDL,只不过系统为我们做了封装从而方便上层调用而已。

2.AIDL跨进程通信流程

1 服务端

服务端首先要创建一个Service用来监听客户端的连接请求,然后创建一个AIDL文件,将暴露给客户端的接口在这个AIDL文件中声明,最后在Service中实现这个接口。

2 客户端

客户端首先需要绑定服务端的Service,绑定成功后,将服务端返回的Binder对象转换成AIDL接口类型,接着就可以调用AIDL中的方法了。

3 AIDL接口的创建
// Book.aidl
package com.zhouzhuo.client;
// Declare any non-default types here with import statements
parcelable Book;


// BookManager.aidl
package com.zhouzhuo.client;

// Declare any non-default types here with import statements
import com.zhouzhuo.client.Book;

interface BookManager {
    List<Book> getBooks();
        void addBook(inout Book book);
}

在AIDL文件中,并不是所有的数据类型都可以使用的,那么AIDL文件支持哪些数据类型呢?

  • 基本数据类型(int,long,char,boolean,double等)
  • Sting 和CharSequence;
  • List: 只支持ArrayList,并且每个元素都必须能够被AIDL支持
  • Map:只支持HashMap,里面每个元素都必须被AIDL支持,包括key和value。
  • Parcelable:所有实现了Parcelable接口的对象
  • AIDL : 使用的AIDL接口本身也可以在AIDL文件中使用。

非常重要的一点:如果AIDL文件中用到了自定义的Parcelable对象,那么必须新建一个同名的AIDL文件,并在其中声明为Parcelable类型。在BookManager.aidl中,我们用到了Book这个类,所以,我们必须要创建Book.aidl。

除此之外,AIDL中除了基本数据类型,其他类型的参数必须标上方向:in,out或者inout,in表示输入参数类型,out表示输出参数类型,inout表示输入输出参数类型。我们要根据实际需要去指定参数类型,不能一概使用out或者inout,因为这在底层实现是有开销的。为了方便AIDL的开发,建议把所有和AIDL相关的文件全部放进同一个包中,这样做的好处是,当客户端是另一个应用时,我们可以直接把整个包复制到客户端工程中。

4 远程服务端Service的实现
package com.zhouzhuo.server.service;

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

import com.zhouzhuo.client.Book;
import com.zhouzhuo.client.BookManager;

import java.util.ArrayList;
import java.util.List;

public class AIDLService extends Service {
    private final String TAG = this.getClass().getSimpleName();
    //包含book对象的list
    private List<Book> mBooks = new ArrayList<>();
    @Override
    public IBinder onBind(Intent intent) {
        return manager;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        Book book = new Book();
        book.setName("Android 开发艺术探索");
        book.setPrice(28);
        mBooks.add(book);
        super.onCreate();
    }
    private BookManager.Stub manager = new BookManager.Stub() {
        @Override
        public List<Book> getBooks() throws RemoteException {
            Log.e("zhouzhuo","invoking getBooks() method,now the list is:"+mBooks.toString());
            if(mBooks!=null){
                return mBooks;
            }
            return new ArrayList<>();
        }

        @Override
        public void addBook(Book book) throws RemoteException {
            synchronized (this){
                if(mBooks == null){
                    mBooks = new ArrayList<>();
                }
                if(book == null){
                    book = new Book();
                    book.setPrice(22222);
                    book.setName("aaa");
                }

                if(!mBooks.contains(book)){
                    mBooks.add(book);
                }
                //打印mBooks列表,观察客户端传过来的值
                Log.e("zhouzhuo","invoking addBooks() method,now the list is:"+mBooks.toString());
            }

        }
    };
}

值得一提的是,AIDL方法是在服务端的Binder线程池中执行的,因此当多个因此当多个客户端同时使用的时候,会存在多个线程同时访问的情形,所有我们要在AIDL方法中处理多线程同步的。

5 客户端的实现

首先,客户端要绑定远程服务,绑定成功后将服务端返回的Binder对象转换成AIDL接口,然后就可以通过这个接口去调用服务端的远程方法了。

package com.zhouzhuo.aidlfour;

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.util.Log;
import android.view.View;
import android.widget.Toast;

import com.zhouzhuo.client.Book;
import com.zhouzhuo.client.BookManager;

import java.util.List;

/**
 * AIDL方式
 */

public class AIDLActivity extends AppCompatActivity {
    //由AIDL文件生成的java类
    private BookManager mBookManager;
    //标记当前与服务端连接的布尔值,false为未连接,true为连接中
    private boolean mBound ;

    //包含Book对象的list
    private List<Book> mBooks;

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

    public void addBook(View view){
        //如果与服务端的连接处于未连接状态,则尝试连接
        if(!mBound){
            attemptToBindService();
            Toast.makeText(this,"当前与服务端处于未连接状态,正在尝试重连,请稍后再试",Toast.LENGTH_SHORT).show();
            return;
        }
        if(mBookManager == null){
            return;
        }
        Book book = new Book();
        book.setName("App研发录In");
        book.setPrice(30);
        try {
            mBookManager.addBook(book);
        } catch (RemoteException e) {
            e.printStackTrace();
        }

    }

    @Override
    protected void onStart() {
        super.onStart();
        if(!mBound){
            attemptToBindService();
        }
    }

    private void attemptToBindService() {
        Intent intent = new Intent();
        intent.setAction("com.zhouzhuo.server.service.AIDLService");
        intent.setPackage("com.zhouzhuo.server");
        bindService(intent,connection, Context.BIND_AUTO_CREATE);

    }

    private ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.d("zhouzhuo","onServiceConnected");
            mBookManager = BookManager.Stub.asInterface(service);
            mBound = true;
            if(mBookManager!=null){
                try {
                    mBooks = mBookManager.getBooks();
                    Log.d("zhouzhuo","mBooks :"+mBooks.toString());
                } catch (RemoteException e) {
                    Log.d("zhouzhuo","eddddd=="+e.toString());
                    e.printStackTrace();
                }

            }

        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.d("zhouzhuo","onServiceDisconnected");

        }
    };

    @Override
    protected void onStop() {
        super.onStop();
        if(mBound){
            unbindService(connection);
            mBound = false;
        }
    }
}
6 项目地址
7 监听对象的添加和移除

假设有这样一种需求:用户不想时不时的去查询图书列表。当新书到的时候,图书馆通知每一个队这本书感兴趣的用户。首先,我们需要提供一个AIDL接口,每个用户都要实现这个接口并且向图书馆申请新书的提醒功能,当然用户也可以随时提醒这种提醒。

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

推荐阅读更多精彩内容