一、IPC进程间通信

IPC是进程间通信方法的统称,Linux IPC包括以下方法,Android的进程间通信主要采用是哪些方法呢?

1. 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允许无亲缘关系进程间的通信;
   2. 信号(Signal):信号是比较复杂的通信方式,用于通知接受进程有某种事件发生,除了用于进程间通信外,进程还可以发送信号给进程本身;linux除了支持Unix早期信号语义函数sigal外,还支持语义符合Posix.1标准的信号函数sigaction(实际上,该函数是基于BSD的,BSD为了实现可靠信号机制,又能够统一对外接口,用sigaction函数重新实现了signal函数);
   3. 报文(Message)队列(消息队列):消息队列是消息的链接表,包括Posix消息队列system V消息队列。有足够权限的进程可以向队列中添加消息,被赋予读权限的进程则可以读走队列中的消息。消息队列克服了信号承载信息量少,管道只能承载无格式字节流以及缓冲区大小受限等缺点。
   4. 共享内存:使得多个进程可以访问同一块内存空间,是最快的可用IPC形式。是针对其他通信机制运行效率较低而设计的。往往与其它通信机制,如信号量结合使用,来达到进程间的同步及互斥。
   5. 信号量(semaphore):主要作为进程间以及同一进程不同线程之间的同步手段。
   6. 套接口(Socket):更为一般的进程间通信机制,可用于不同机器之间的进程间通信。起初是由Unix系统的BSD分支开发出来的,但现在一般可以移植到其它类Unix系统上:Linux和System V的变种都支持套接字。

二、AIDL(Android Interface Definition Language)

AIDL - Android Interface Definition Language - Android 接口定义语言。因为在Android中,应用程序运行在各自独立的进程里。应用程序之间是不能访问对方的内存空间的。有时为了实现进程间的通信,要用到PCI机制。Android支持PCI机制,但是需要Android能读懂的序列化数据(marshaling/un marshaling of data).    AIDL就是为了描述这样的数据产生的,它是一种接口定义语言。语法类似JAVA,.aidl文件里面写的是公布给客户端接口声明。

本文中,为了吸取客户端,服务器端两个android 项目的经验,将两段整合到一个android项目里,这样可能会比较清晰。下面就看看实例解析吧。

三、AIDL介绍及解析

1.AIDL介绍

在Android中,每个应用(Application)执行在它自己的进程中,无法直接调用到其他应用的资源,这也符合“沙箱”的理念。所谓沙箱原理,一般来说用在移动电话业务中,简单地说旨在部分地或全部地隔离应用程序。关于沙箱技术我们这里就不多做介绍了。因此,在Android中,当一个应用被执行时,一些操作是被限制的,比如访问内存,访问传感器,等等。这样做可以最大化地保护系统,免得应用程序“为所欲为”。那我们有时需要在应用间交互,怎么办呢?于是,Android需要实现IPC协议。然而,这个协议还是有点复杂,主要因为需要实现数据管理系统(在进程或线程间传递数据)。Android为我们实现了自己的IPC,也就是AIDL。

2.定义AIDL接口

AIDL是IPC的一个轻量级实现,用了对于Java开发者来说很熟悉的语法。Android也提供了一个工具,可以自动创建Stub(类构架,类骨架)。当我们需要在应用间通信时,我们需要按以下几步走:

a. 定义一个AIDL接口

b. 为远程服务(Service)实现对应Stub

c. 将服务“暴露”给客户程序使用

3.实例解析

AIDL的语法很类似Java的接口(Interface),只需要定义方法的签名。

AIDL支持的数据类型与Java接口支持的数据类型有些不同

a. 所有基础类型(int, char, 等)

b. String,List,Map,CharSequence等类

c. 其他AIDL接口类型

d. 所有Parcelable的类

下面以一个加法器为例来做解析。

(1)创建工程,创建AIDL接口:新建一个文件,命名为IAdditionService.aidl 即可

package com.czm.hellosumaidl;

interface IAdditionService{
int add(int value1,int value2);
}

一旦文件被保存,Android的AIDL工具会在 gen/com/android/hellosumaidl 这个文件夹里自动生成对应的 IAdditionService.java 这个文件。因为是自动生成的,所以无需改动。这个文件里就包含了 Stub ,我们接下来要为我们的远程服务实现这个Stub。

(2)实现远程服务

首先我们新建一个类,取名叫AdditionService.java 。为了 实现我们 的服务,我们需要让 这个 类中的 onBind方法返回一个 IBinder 类的对象。这个 IBinder 类的对象就代表了远程服务的实现。为了实现这个服务,我们要用到自动生成的子类 IAdditionService.Stub 。在其中,我们也必须实现我们之前在AIDL文件中定义的 add() 函 数。下面是我们远程服务的代码:

public class AdditionService extends Service{

    @Override
public void onCreate() {
super.onCreate();
} @Override
public IBinder onBind(Intent intent) {
// TODO Auto-generated method stub
return new IAdditionService.Stub() { @Override
public int add(int value1, int value2) throws RemoteException {
// TODO Auto-generated method stub
return value1 + value2;
}
};
} @Override
public void onDestroy() {
super.onDestroy();
} }

(3)提供服务(“暴露”服务给使用者)

一旦实现了服务中的onBind方法 ,我们就可以把客户程序(在这里是MainActivity.java )与服务连接起来了。为了建立这样的一个链接 ,我们需要实现 ServiceConnection 类。我们在 MainActivity.java 创建一个内部类 AdditionServiceConnection ,这个类继承 ServiceConnection 类,并且重写了它的两个方法: onServiceConnected 和 onServiceDisconnected。下面给出 内部类的代码:

class AdditionServiceConnection implements ServiceConnection {

        @Override
public void onServiceConnected(ComponentName name, IBinder boundService) {
// TODO Auto-generated method stub
service = IAdditionService.Stub.asInterface((IBinder) boundService);
Toast.makeText(MainActivity.this, "Service connected",
Toast.LENGTH_SHORT).show();
} @Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
service = null;
Toast.makeText(MainActivity.this, "Service disconnected",
Toast.LENGTH_SHORT).show();
} }

(4)相关代码及效果截图

MainActivity.java的全部代码如下:

package com.czm.hellosumaidl;

import android.app.Activity;
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.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast; public class MainActivity extends Activity { IAdditionService service;
AdditionServiceConnection connection;
EditText value1;
EditText value2;
TextView result;
Button buttonCalc;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
initService(); buttonCalc = (Button) findViewById(R.id.buttonCalc); value1 = (EditText) findViewById(R.id.value1);
value1.setText("23");
value2 = (EditText) findViewById(R.id.value2);
value2.setText("24");
result = (TextView) findViewById(R.id.result);
buttonCalc.setOnClickListener(new OnClickListener() { @Override
public void onClick(View v) {
int v1, v2, res = -1;
v1 = Integer.parseInt(value1.getText().toString());
v2 = Integer.parseInt(value2.getText().toString()); try {
res = service.add(v1, v2);
} catch (RemoteException e) {
e.printStackTrace();
} result.setText(Integer.valueOf(res).toString());
}
});
} @Override
protected void onDestroy() {
super.onDestroy();
releaseService();
} /*
* This function connects the Activity to the service
*/
private void initService() {
connection = new AdditionServiceConnection(); Intent i = new Intent();
i.setClassName("com.czm.hellosumaidl",com.czm.hellosumaidl.AdditionService.class.getName());
boolean ret = this.bindService(i, connection, Context.BIND_AUTO_CREATE);
} /*
* This function disconnects the Activity from the service
*/
private void releaseService() {
unbindService(connection);
connection = null;
} class AdditionServiceConnection implements ServiceConnection { @Override
public void onServiceConnected(ComponentName name, IBinder boundService) {
// TODO Auto-generated method stub
service = IAdditionService.Stub.asInterface((IBinder) boundService);
Toast.makeText(MainActivity.this, "Service connected",
Toast.LENGTH_SHORT).show();
} @Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
service = null;
Toast.makeText(MainActivity.this, "Service disconnected",
Toast.LENGTH_SHORT).show();
} } }

效果截图如下:

          

执行AIDL接口之前                                                                                                     执行AIDL接口之后

Android开发之IPC进程间通信-AIDL介绍及实例解析的更多相关文章

  1. Android开发之UI更新交互机制与实例解析

    android开发过程中,经常需要更新UI的状态和文案等.这是就需要对UI进行 更新.在android中更新UI一般有三种方法,handler机制.RunOnUiThread方法以及AsyncTask ...

  2. Android开发之InstanceState详解

    Android开发之InstanceState详解   本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...

  3. 【Android UI】Android开发之View的几种布局方式及实践

    引言 通过前面两篇: Android 开发之旅:又见Hello World! Android 开发之旅:深入分析布局文件&又是“Hello World!” 我们对Android应用程序运行原理 ...

  4. Android开发之旅: Intents和Intent Filters(理论部分)

    引言 大部分移动设备平台上的应用程序都运行在他们自己的沙盒中.他们彼此之间互相隔离,并且严格限制应用程序与硬件和原始组件之间的交互. 我们知道交流是多么的重要,作为一个孤岛没有交流的东西,一定毫无意义 ...

  5. Android开发之Java必备基础

    Android开发之Java必备基础 Java类型系统 Java语言基础数据类型有两种:对象和基本类型(Primitives).Java通过强制使用静态类型来确保类型安全,要求每个变量在使用之前必须先 ...

  6. [置顶] Android开发之MediaPlayerService服务详解(一)

    前面一节我们分析了Binder通信相关的两个重要类:ProcessState 和 IPCThreadState.ProcessState负责打开Binder 驱动,每个进程只有一个.而 IPCThre ...

  7. Android 开发之旅:深入分析布局文件&又是“Hello World!”

    http://www.cnblogs.com/skynet/archive/2010/05/20/1740277.html 引言 上篇可以说是一个分水岭,它标志着我们从Android应用程序理论进入实 ...

  8. [置顶] Android开发之serviceManager分析

    Android 开发之serviceManager分析 在Android系统中用到最多的通信机制就是Binder,Binder主要由Client.Server.ServiceManager和Binde ...

  9. Android开发之TextView高级应用

    Android开发之TextView高级应用 我们平时使用TextView往往让它作为一个显示文字的容器,但TextView的功能并不局限于此.以下就和大家分享一下TextView的一些使用技巧. A ...

随机推荐

  1. jstring 和char 之间的转换方法

    //jstring to char* char* jstringTostring(JNIEnv* env, jstring jstr) { char* rtn = NULL; jclass clsst ...

  2. Windows系统性能提升方法

    看前提醒:在确认没有病毒和流氓软件的前提下,建议优化电脑:以下操作已经在Win7上试验,Win7以上的园友自己试验,自己感受,对电脑无害,但操作时请务必小心 设置虚拟内存 虚拟内存最小值物理内存1.5 ...

  3. 【CSS3】

    Web前端实验室http://demo.doyoe.com/ ::before ::afterCSS3已经将伪元素的前缀更改为双冒号,而伪类则保持为单冒号 backface-visibility ht ...

  4. Dell笔记本禁用触摸板的方法

    一·找到触摸板驱动所在的文件夹(其他型号 的本本,请自己探索一下,找到驱动在哪就行),一般 在 C:\program files\delltpad 中(若没有请下载安 装 ),如图: 二·双击 Del ...

  5. jQuery插件之-Poshy Tip

    jQuery插件Poshy Tip是一个强大的jQuery Tooltips插件,它有多种不同的外观.同时可以作为 Form Tooltips使用,并且可以自定义气泡出现的位置.在处理表单验证提示上能 ...

  6. [WebKit]浏览器的加载与页面性能优化

    非常棒.非常系统的一份资料,值得阅读! 原文来自百度泛用户体验. 作者:nwind 本文将探讨浏览器渲染的loading过程,主要有2个目的: 了解浏览器在loading过程中的实现细节,具体都做了什 ...

  7. 利用URLRewriter重写url地址

    首先,当然是下载URLRewriter了 download.microsoft.com/download/0/4/6/0463611e-a3f9-490d-a08c-877a83b797cf/MSDN ...

  8. JS中document.createElement()用法及注意事项

    今天处理了一个日期选择器的ie和ff的兼容问题,本来这种情况就很难找错误,找了好久才把错误定位到js中创建元素的方法document.createElement(),这个方法在ie下支持这样创建元素 ...

  9. windows log

    http://technet.microsoft.com/zh-CN/sysinternals http://technet.microsoft.com/en-us/sysinternals/bb89 ...

  10. CentOS6.x安装配置nginx [转]

    博文来源:http://leyewen.blog.163.com/   nginx安装 nginx的官网:http://nginx.org/   相应下载页面:http://nginx.org/en/ ...