在Android开发中,有时会用到多进程通信,这时,可选的方案为:

  1. Bundle    :四大组件之间的进程间通信

  2. 文件共享   :适合无并发情景

  3. Messager : 低并发的一对多即使通信,无RPC需求或无须要返回结果的RPC需求

  4. AIDL        :一对多通信且有RPC需求

  5. Content Provider : 一对多的进程间数据共享

  6. Socket     : 网络数据交换

  这里使用AIDL来举例,其实,Message底层也是AIDL来处理的,是对AIDL的一个简单封装,AIDL用到一个Android系统的核心--Binder,Binder在Android系统中用来处理进程间通信,它基于OpenBinder来实现的,是一个类似于COM和CORBA的分布式组件架构,其本身也挺复杂的,源码还没看懂,一开始看时感觉云里雾里的,但是,先用着吧,实践帮助理论,二者不断交错,肯定有一天弄懂的,哈哈~

  在一个APP应用里,通过process属性可以方便测试多进程通信,这里采用另一种方式,使用两个APP应用,一个是服务端,一个是客户端,其实,两种方式都一样道理,因为对于同一个应用的不同组件,如果它们运行在不同的进程中,它们与分别属于两个应用没有什么本质区别。

  开始测试:

  1. 新建服务端项目:TestAIDLServer:

  

  2. 新建一个用户管理类User.java,AIDL需要使用Paracelable来序列化和反序列化对象:

package cn.linjk.testaidlserver;

import android.os.Parcel;
import android.os.Parcelable; /**
* Created by LinJK on 06/12/2016.
*/ public class User implements Parcelable{ public int userId;
public String userName; public User(int userId, String userName) {
this.userId = userId;
this.userName = userName;
} @Override
public int describeContents() {
return 0;
} @Override
public void writeToParcel(Parcel pParcel, int pI) {
pParcel.writeInt(userId);
pParcel.writeString(userName);
} public static final Parcelable.Creator<User> CREATOR = new Parcelable.Creator<User>() {
@Override
public User createFromParcel(Parcel pParcel) {
return new User(pParcel);
} @Override
public User[] newArray(int pI) {
return new User[pI];
}
}; private User(Parcel in) {
userId = in.readInt();
userName = in.readString();
}
}

  3. 新建AIDL文件夹并新建aidl接口文件:

    3.1 新建aidl文件夹:

      

      新建完成后目录结构如下:

      

    3.2 新建aidl接口文件,在接口文件用到了User类,也需要新建对应的类的aidl文件来声明类,并且在接口文件需要显示import这个User类

      

      这里声明了两个方式:新增用户和获取当前所有用户的方法。

    (小插曲:

      我们在这里make一次工程后,会生成一个java文件:

      

      使用aidl来处理,我们就不用自己写Binder通信,当然,也可以手写,这样有利于更加理解binder的原理,但是初学者就麻烦了,所有,两者都可以实现,AIDL文件的本质就是android系统为我们提供了一种快速实现binder的工具而已。

      )

  4. 为了方便AIDL开发,所有和AIDL相关的类和文件放入同一个包内,如果类的完整路径不一样,在反序列化时会失败,开发用户端时直接把这个包复制到客户端应用即可,这里把User类放进与aidl同级目录下。

  5. 服务新建一个Service,实现AIDL接口方法(这里为了方便,先在服务端准备两条数据):

package cn.linjk.testaidlserver;

import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.os.RemoteException;
import android.support.annotation.Nullable; import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList; import cn.linjk.testaidlserver.aidl.IUserManager;
import cn.linjk.testaidlserver.aidl.User; /**
* Created by LinJK on 06/12/2016.
*/ public class ServiceUserManager extends Service{ private static final String TAG = ServiceUserManager.class.getSimpleName(); private CopyOnWriteArrayList<User> mUserCopyOnWriteArrayList = new CopyOnWriteArrayList<>(); private Binder mBinder = new IUserManager.Stub() {
@Override
public void addUser(User user) throws RemoteException {
mUserCopyOnWriteArrayList.add(user);
} @Override
public List<User> getAllUsers() throws RemoteException {
return mUserCopyOnWriteArrayList;
}
}; @Override
public void onCreate() {
super.onCreate();
mUserCopyOnWriteArrayList.add(new User(1, "Jim"));
mUserCopyOnWriteArrayList.add(new User(2, "Jim_LinJK"));
} @Nullable
@Override
public IBinder onBind(Intent pIntent) {
return mBinder;
}
}

    6. 启动服务端,发现出现如下错误,找不到数据类?

    

    修改app/build.gradle文件配置:

    

    7. 可以了,服务端开启完成。

    8. 新建客户端项目:

    

    9. 把服务端的src/main/aidl文件下的文件拷贝到客户端同样目录路径下(如果同一个APP的话就不用拷贝了,共用的),完成后如下:

    

    10. 编写连接服务端代码:

    因为这里需要连接服务端的servicemanager,所以需要把ServiceUserManager类也移到aidl文件夹下面,并保证服务端项目和客户端项目文件和其内容一致,最后aidl文件夹下文件为:

    

package cn.linjk.testaidlclient;

import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.IBinder;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log; import java.util.List; import cn.linjk.testaidlserver.aidl.IUserManager;
import cn.linjk.testaidlserver.aidl.ServiceUserManager;
import cn.linjk.testaidlserver.aidl.User; public class MainActivity extends AppCompatActivity {
private final static String TAG = MainActivity.class.getSimpleName(); @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); bindService(new Intent(this, ServiceUserManager.class), mServiceConnection, Context.BIND_AUTO_CREATE);
} @Override
protected void onDestroy() {
unbindService(mServiceConnection);
super.onDestroy();
} private ServiceConnection mServiceConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName pComponentName, IBinder pIBinder) {
IUserManager lvIUserManager = IUserManager.Stub.asInterface(pIBinder); try {
List<User> lvUserList = lvIUserManager.getAllUsers();
for (User lvUser : lvUserList) {
Log.i(TAG, "[User name] : " + lvUser.userName + " and [User id] : " + lvUser.userId);
}
}
catch (Exception e) {
e.printStackTrace();
}
} @Override
public void onServiceDisconnected(ComponentName pComponentName) {
//
}
};
}

    11. 先启动服务端APP

     12. 再启动客户端APP,可以看到,客户端APP能获取服务端APP的数据:

    13. 在客户端APP增加一个按钮,用于向服务端新增一条用户数据,界面代码比较简单,就不写了,看看按钮的监听部分代码:

btnAddUser.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View pView) {
try {
User lvUser = new User(3, "Apple");
lvIUserManager.addUser(lvUser);
//
List<User> lvUserList = lvIUserManager.getAllUsers();
for (User lvUserTmp : lvUserList) {
Log.i(TAG, "[User name] : " + lvUserTmp.userName + " and [User id] : " + lvUserTmp.userId);
}
}
catch (Exception e) {
e.printStackTrace();
}
}
});

    14. 再次运行客户端APP,看看日志:

    一开始没有id为3的用户,点击按钮后,再次看看用户list,新增了一个用户成功:    

    到这里,aidl的测试已经可以了,当然,整个逻辑还有很多异常没考虑,这里就不全部完善了,例如不用每次从服务器查询数据,而是服务器数据更新了自动提醒客户端进行界面刷新功能等。

    代码已提交GitHub: https://github.com/linjk/TestAIDL.git

Android开发之---AIDL的更多相关文章

  1. Android开发了解——AIDL

    AIDL:Android Interface Definition Language,即Android接口定义语言. 什么是AIDL Android系统中的进程之间不能共享内存,因此,需要提供一些机制 ...

  2. Android开发-解决 AIDL 中找不到couldn't find import for class错误

    最近在使用AIDL做IPC的时候,在处理复杂的数据类型的时候,编译器总是报couldn't find import for class错误,所以在这里总结下AIDL使用的时候的一些注意事项,希望对你能 ...

  3. Android开发-API指南-AIDL

    Android Interface Definition Language (AIDL) 英文原文:http://developer.android.com/guide/components/aidl ...

  4. Android开发,Eclipse创建aidl接口时,出错

    Android开发中,当我们需要调用远程Service时,我们一般通过远程接口(RMI)来实现的,而Android的RMI需要AIDL(Android Interface Definition Lan ...

  5. android开发系列之aidl

    aidl在android开发中的主要作用就是跨进程通讯来着,说到进程相比很多人都是非常熟悉了,但是为什么会有跨进程通讯这个概念呢?原来在android系统中,有这么一套安全机制,为了各个Apk数据的独 ...

  6. Android开发——进程间通信之AIDL(二)

    0.  前言 不论是Android还是其它操作系统.都会有自己的IPC机制.所谓IPC(Inter-Process Communication)即进程间通信.首先线程和进程是非常不同的概念,线程是CP ...

  7. Qt for Android开发环境搭建及测试过程记录

    最近学习了Qt的QML编程技术,感觉相较于以前的QtGUI来说更方便一些,使用QML可以将界面与业务逻辑解耦,便于开发. QML支持跨平台,包括支持Android平台,因此可以使用Qt的QML进行An ...

  8. Android开发学习清单

    目录: 第1章 Android应用与开发环境1.1 Android的发展和历史1.1.1 Android的发展和简介1.1.2 Android平台架构及特性1.2 搭建Android开发环境1.2.1 ...

  9. Android开发权威指南(第2版)新书发布

    <Android 开发权威指南(第二版)>是畅销书<Android开发权威指南>的升级版,内容更新超过80%,是一本全面介绍Android应用开发的专著,拥有45 章精彩内容供 ...

随机推荐

  1. Android BLE 蓝牙编程(四)

    接上篇,我们已经实现了短震,长震的功能了- 现在我们需要实现点击后一直震动的功能 开始我的想法是再循环中不断执行write方法,然而这个办法行不通. 系统会报错. 那要如何实现这个想法呢?其实很简单, ...

  2. HTTP Status

    Web服务器响应浏览器或其他客户程序的请求时,其应答一般由以下几个部分组成:一个状态行,几个应答 头,一个空行,内容文档.下面是一个最简单的应答 : 状态行包含HTTP版本.状态代码.与状态代码对应的 ...

  3. Redis JedisPool

    获取连接池,通常连接池为单例,这里使用 双端检测机制保证只有一个实例 public class JedisPoolUtil { private static volatile JedisPool je ...

  4. matlab进阶:常用功能的实现,常用函数的说明

    常用功能的实现 获取当前脚本所在目录 current_script_dir = fileparts(mfilename('fullpath')); % 结尾不带'/' 常用函数的说明 bsxfun m ...

  5. Linux 下Nginx编译安装

    Untitled .note-content {font-family: 'Helvetica Neue', Arial, 'Hiragino Sans GB', STHeiti, 'Microsof ...

  6. 业务中是否有必要让所有的ViewController统一继承抽象类

    疑问来自:这里 1.事出有因 其中博主说道的情况我其实也经历过,当时还在找到一个模式可以改变这样的情况.直到有一天看到这个博客,今天晚上有时间来规整一下博主的思路和写了一个测试代码. 这是我目前的Ap ...

  7. 问你觉得iOS7为什么要扁平化,扁平化和之前的比有什么优势

    问你觉得iOS7为什么要扁平化,扁平化和之前的比有什么优势 苹果首席设计师谈为何会在iOS上选择扁平风格http://ndnews.oeeee.com/html/201306/11/71078.htm ...

  8. 网站访问量大 怎样优化mysql数据库

    MySQL优化的一些建议,单机MySQL的优化我分为三个部分,一是服务器物理硬件的优化,二是 MySQL安装时的编译优化,三是自身配置文件my.cnf的优化:如果单机的优化也解决不了你的数据库的压力的 ...

  9. yuv420转rgb 及 rgb转bmp保存

    /// <summary> /// 将一桢 YUV 格式的图像转换为一桢 RGB 格式图像. /// </summary> /// <param name="y ...

  10. CentOS添加163源

    1.备份/etc/yum.repos.d/CentOS-Base.repo mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-B ...