android中的AIDL进程间通信示例

 更新时间:2016年11月05日 15:50:04   作者:VincentCZW  
进程之间不能共享内存,那么怎么在不同的应用程序中进行通讯,这就要依赖AIDL机制,本文详细介绍了android中的AIDL进程间通信示例,有兴趣的可以了解一下。

关于IPC应该不用多介绍了,Android系统中的进程之间不能共享内存,那么如果两个不同的应用程序之间需要通讯怎么办呢?比如公司的一个项目要更新,产品的需求是依附于当前项目开发一个插件,但是呢这个插件功能以及界面比较复杂,不能和当前项目在一个进程中,同时呢,还要用到当前项目中已经写好了的一些东西,那么因为新开发的依附于当前项目的插件和当前项目不是一个进程,因此不能共享内存,就出现了问题,于是,需要提供一些机制在不同进程之间进行数据通信,这个机制就是AIDL了。

一、一个android中AIDL的简单例子

假如是这样,现在有一个项目中提供了比较成熟的计算的方法,而现在我想开发一款软件其中一个模块想用到一个计算类,而我又不想重新写了,那么就可以通过AIDL实现啦。假设,已经开发完成的那个已经提供了比较成熟的计算类的程序叫AIDLCalculateDemoServer(相当于服务器),而我要写的程序叫AIDLCalculateDemoClient(相当于客户端),类似与客户端服务器模式。首先至关的看下工程结构图:
                                     图1-1 服务器                                                       图1-2 客户端 

现在假设自己写的程序要调用服务端的运算界面,输入num1和num2,进行远程运算,调用服务端的接口,服务端运算好之后,返回结果给客户端,效果图如下:
然后来看看实现,首先需要定义AIDL接口,客户端和服务器端都要定义,并且要在同一包中,也就是图1-1和图1-2 com.example.aidl.calculate中的CalculateInterface,其中的代码如下:

package com.example.aidl.calculate;

interface CalculateInterface {
  double doCalculate(double a, double b);
 }

编译发现,目录结构如图1-1和图1-2中gen/com.example.aidl.calculate中多了CalculateInterface.java文件,内容如下:

 package com.example.aidl.calculate;
 
interface CalculateInterface {
   double doCalculate(double a, double b);
 }

定义好接口就是要看服务端和客户端的代码啦,其中服务端主要看CalculateService代码,这个一个继承Service的类,在其中对AIDL中的接口进行赋予实际意义,如下:

 package com.example.calculate;

import com.example.aidl.calculate.CalculateInterface;
import com.example.aidl.calculate.CalculateInterface.Stub;

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

public class CalculateService extends Service {
  
  private static final String              TAG            =  "CalculateService";
  
  @Override
  public IBinder onBind(Intent arg0) {
    // TODO Auto-generated method stub
    logE("onBind()");
    return mBinder;
  }
  
  @Override
  public void onCreate() {
    // TODO Auto-generated method stub
    logE("onCreate()");
    super.onCreate();
  }
  
  @Override
  public void onStart(Intent intent, int startId) {
    // TODO Auto-generated method stub
    logE("onStart()");
    super.onStart(intent, startId);
  }
  
  @Override
  public boolean onUnbind(Intent intent) {
    // TODO Auto-generated method stub
    logE("onUnbind()");
    return super.onUnbind(intent);
  }
  
  @Override
  public void onDestroy() {
    // TODO Auto-generated method stub
    logE("onDestroy()");
    super.onDestroy();
  }
  
  private static void logE(String str) {
    Log.e(TAG, "--------" + str + "--------");
  }
  
  private final CalculateInterface.Stub mBinder = new CalculateInterface.Stub() {
    
    @Override
    public double doCalculate(double a, double b) throws RemoteException {
      // TODO Auto-generated method stub
      Log.e("Calculate", "远程计算中");
      Calculate calculate = new Calculate();
      double answer = calculate.calculateSum(a, b);
      return answer;
    }
  };
}

然后可以看看,关键的服务都提供完毕,那么在客户端是怎么访问的呢,要进行绑定服务和一个ServiceConnection类完成,如下:

package com.example.calculate;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.graphics.Color;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;

import com.example.aidl.calculate.CalculateInterface;
import com.example.aidlcalculatedemoclient.R;

public class CalculateClient extends Activity {
  private static final String         TAG            =      "CalculateClient";
  
  private       Button         btnCalculate;
  
  private       EditText        etNum1;
  
  private         EditText        etNum2;
  
  private       TextView        tvResult;
  
  private        CalculateInterface   mService;
  
  private       ServiceConnection    mServiceConnection = new ServiceConnection() {
    
    @Override
    public void onServiceDisconnected(ComponentName name) {
      // TODO Auto-generated method stub
      logE("disconnect service");
      mService = null;
    }
    
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
      // TODO Auto-generated method stub
      logE("connect service");
      mService = CalculateInterface.Stub.asInterface(service);
    }
  };
  
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    // TODO Auto-generated method stub
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    
    Bundle args = new Bundle();
    Intent intent = new Intent("com.example.calculate.CalculateService");
    intent.putExtras(args);
    bindService(intent, mServiceConnection, Context.BIND_AUTO_CREATE);
    
    etNum1 = (EditText) findViewById(R.id.et_num_one);
    etNum2 = (EditText) findViewById(R.id.et_num_two);
    
    tvResult = (TextView) findViewById(R.id.tv_result);
    
    btnCalculate = (Button) findViewById(R.id.btn_cal);
    
    btnCalculate.setOnClickListener(new View.OnClickListener() {
      
      @Override
      public void onClick(View v) {
        // TODO Auto-generated method stub
        
        logE("开始远程运算");
        try {
          double num1 = Double.parseDouble(etNum1.getText().toString());
          double num2 = Double.parseDouble(etNum2.getText().toString());
          String answer = "计算结果:" + mService.doCalculate(num1, num2);
          tvResult.setTextColor(Color.BLUE);
          tvResult.setText(answer);
              
        } catch (RemoteException e) {
        }
      }
    });
  }
  
  private void logE(String str) {
    Log.e(TAG, "--------" + str + "--------");
  }
}

如此一来,大功已经基本告成,最后,我们在来看看服务端的配置文件吧:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.example.aidlcaculatedemoserver"
  android:versionCode="1"
  android:versionName="1.0" >

  <uses-sdk
    android:minSdkVersion="8"
    android:targetSdkVersion="17" />

  <application
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
      android:name="com.example.aidlcaculatedemoserver.MainActivity"
      android:label="@string/app_name" >
      <intent-filter>
        <action android:name="android.intent.action.MAIN" />

        <category android:name="android.intent.category.LAUNCHER" />
      </intent-filter>
    </activity>
    <service android:name="com.example.calculate.CalculateService">
      <intent-filter>
        <action android:name="com.example.calculate.CalculateService" />
      </intent-filter>
    </service>
  </application>

</manifest>

二、写AIDL注意事项

1. 客户端和服务端的AIDL接口文件所在的包必须相同

2. 需要一个Service类的配合

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Android自定义控件仿QQ抽屉效果

    Android自定义控件仿QQ抽屉效果

    这篇文章主要为大家详细介绍了Android自定义控件仿QQ抽屉效果的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • Android drawFunctor 原理及应用详情

    Android drawFunctor 原理及应用详情

    这篇文章主要介绍了Android drawFunctor原理及应用详情,drawFunctor是Android提供的一种在RenderThread渲染流程中插入执行代码机制,更多相关内容需要的小伙伴可以参考一下
    2022-08-08
  • Android开发自学笔记(四):APP布局下

    Android开发自学笔记(四):APP布局下

    这篇文章主要介绍了Android开发自学笔记(四):APP布局下,本文是上一篇的补充,需要的朋友可以参考下
    2015-04-04
  • Android编程实现的短信编辑器功能示例

    Android编程实现的短信编辑器功能示例

    这篇文章主要介绍了Android编程实现的短信编辑器功能,涉及Android权限控制、界面布局及短信功能相关操作技巧,需要的朋友可以参考下
    2017-08-08
  • 往Android系统中添加服务的方法教程

    往Android系统中添加服务的方法教程

    最近因为平台升级,需要在系统中添加一些服务,所以将整个过程总结一下,下面这篇文章主要给大家介绍了往Android系统中添加服务的方法教程,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-05-05
  • Android编程ViewPager回弹效果实例分析

    Android编程ViewPager回弹效果实例分析

    这篇文章主要介绍了Android编程ViewPager回弹效果,以实例形式较为详细的分析了ViewPager回弹效果的相关使用技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-10-10
  • Android获取清单文件中的meta-data,解决碰到数值为null的问题

    Android获取清单文件中的meta-data,解决碰到数值为null的问题

    这篇文章主要介绍了Android获取清单文件中的meta-data,解决碰到数值为null的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-03-03
  • Android实现QQ侧滑(删除、置顶等)功能

    Android实现QQ侧滑(删除、置顶等)功能

    这篇文章主要为大家详细介绍了Android实现QQ侧滑删除、置顶等功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • Android修改源码解决Alertdialog触摸对话框边缘消失的问题

    Android修改源码解决Alertdialog触摸对话框边缘消失的问题

    在开发的时候遇到一个问题,就是一触摸对话框边缘外部,对话框会自动消失。这个问题很纠结啊,查找了一下发现从Android 4.0开始,AlertDialog有了变化,就是在触摸对话框边缘外部,对话框会自动消失,查了源码,找到解决办法如下
    2013-11-11
  • Android编程实现动态支持多语言的方法

    Android编程实现动态支持多语言的方法

    这篇文章主要介绍了Android编程实现动态支持多语言的方法,涉及Android资源、控件及属性相关操作技巧,需要的朋友可以参考下
    2017-06-06

最新评论