android基础---->aidl服务的使用

  AIDL和其他的IDL类似,它允许你定义程序接口,以便客户端与服务器端通过IPC机制交互。在android上面,一个进程一般不能访问另外进程的内存。因此,Android平台将这些跨进程访问的对象分解成操作系统能够识别的简单对象。并为跨应用访问而特殊编排和整理这些对象。用于编排和整理这些对象的代码编写起来十分冗长,所以Android的AIDL提供了相关工具来自动生成这些代码。今天,我们开始AIDL的学习。

简单说明

一,AIDL的定义:

它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口

二、AIDL的开发流程:

  • 定义AIDL文件(先在服务端定义,客户端也要定义,内容跟服务端一样)
  • 服务端实现接口
  • 客户端调用接口

具体的aidl的说明:  http://www.cnblogs.com/popapa/p/android_aidl.html

AIDL的简单使用之定义 AIDL 接口

为了测试,我写了aidl服务器端和aidl客户端!

aidl服务器端

项目结构如下:

一、 创建.aidl文件:IRemoteAIDL.aidl

// IRemoteAIDL.aidl
package com.huhx.linux;

// Declare any non-default types here with import statements

interface IRemoteAIDL {
    int add(int a, int b);
}

二、 定义服务类:RemoteService.java

package com.huhx.linux.aidlserver;

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

import com.huhx.linux.IRemoteAIDL;

public class RemoteService extends Service {
    private final static String TAG = "RemoteService";

    private IRemoteAIDL.Stub binder = new IRemoteAIDL.Stub() {
        @Override
        public int add(int a, int b) throws RemoteException {
            Log.i(TAG, "a = " + a + ", b = " + b);
            return a + b;
        }
    };

    @Override
    public IBinder onBind(Intent intent) {
        return binder;
    }
}

三、 在AndroidManifest.xml文件中声明服务:

<service android:name=".RemoteService" android:enabled="true" android:exported="true" />

aidl客户端

项目结构如下:

一、 创建与服务器端一样的aidl文件: IRemoteAIDL.aidl

// IRemoteAIDL.aidl
package com.huhx.linux;

// Declare any non-default types here with import statements

interface IRemoteAIDL {
    int add(int a, int b);
}

二、 在MainActivity中,得到远程服务并调用它的方法。

package com.huhx.linux.aidlclient;

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

import com.huhx.linux.IRemoteAIDL;

public class MainActivity extends AppCompatActivity {
    private IRemoteAIDL iRemoteAIDL;

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

    private ServiceConnection conn = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iRemoteAIDL = IRemoteAIDL.Stub.asInterface(service);
        }
        @Override
        public void onServiceDisconnected(ComponentName name) {
            iRemoteAIDL = null;
        }
    };

    // 绑定服务
    private void binServices() {
        Intent intent = new Intent();
        intent.setComponent(new ComponentName("com.huhx.linux.aidlserver", "com.huhx.linux.aidlserver.RemoteService"));
        bindService(intent, conn, Context.BIND_AUTO_CREATE);
    }

    // 处理相加的方法
    public void addMethod(View view) {
        try {
            int result = iRemoteAIDL.add(3, 4);
            Toast.makeText(MainActivity.this, "3 + 4 = " + result, Toast.LENGTH_SHORT).show();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
}

三、 根据aidl文件,android生成的接口如下(可以不关注):

/*
 * This file is auto-generated.  DO NOT MODIFY.
 * Original file: J:\\android\\andridtest\\andridtest\\2016-02-29\\BaseTest8\\aidlclient\\src\\main\\aidl\\com\\huhx\\linux\\IRemoteAIDL.aidl
 */
package com.huhx.linux;
// Declare any non-default types here with import statements

public interface IRemoteAIDL extends android.os.IInterface
{
/** Local-side IPC implementation stub class. */
public static abstract class Stub extends android.os.Binder implements com.huhx.linux.IRemoteAIDL
{
private static final java.lang.String DESCRIPTOR = "com.huhx.linux.IRemoteAIDL";
/** Construct the stub at attach it to the interface. */
public Stub()
{
this.attachInterface(this, DESCRIPTOR);
}
/**
 * Cast an IBinder object into an com.huhx.linux.IRemoteAIDL interface,
 * generating a proxy if needed.
 */
public static com.huhx.linux.IRemoteAIDL asInterface(android.os.IBinder obj)
{
if ((obj==null)) {
return null;
}
android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
if (((iin!=null)&&(iin instanceof com.huhx.linux.IRemoteAIDL))) {
return ((com.huhx.linux.IRemoteAIDL)iin);
}
return new com.huhx.linux.IRemoteAIDL.Stub.Proxy(obj);
}
@Override public android.os.IBinder asBinder()
{
return this;
}
@Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException
{
switch (code)
{
case INTERFACE_TRANSACTION:
{
reply.writeString(DESCRIPTOR);
return true;
}
case TRANSACTION_add:
{
data.enforceInterface(DESCRIPTOR);
int _arg0;
_arg0 = data.readInt();
int _arg1;
_arg1 = data.readInt();
int _result = this.add(_arg0, _arg1);
reply.writeNoException();
reply.writeInt(_result);
return true;
}
}
return super.onTransact(code, data, reply, flags);
}
private static class Proxy implements com.huhx.linux.IRemoteAIDL
{
private android.os.IBinder mRemote;
Proxy(android.os.IBinder remote)
{
mRemote = remote;
}
@Override public android.os.IBinder asBinder()
{
return mRemote;
}
public java.lang.String getInterfaceDescriptor()
{
return DESCRIPTOR;
}
@Override public int add(int a, int b) throws android.os.RemoteException
{
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
int _result;
try {
_data.writeInterfaceToken(DESCRIPTOR);
_data.writeInt(a);
_data.writeInt(b);
mRemote.transact(Stub.TRANSACTION_add, _data, _reply, 0);
_reply.readException();
_result = _reply.readInt();
}
finally {
_reply.recycle();
_data.recycle();
}
return _result;
}
}
static final int TRANSACTION_add = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
}
public int add(int a, int b) throws android.os.RemoteException;
}

IRemoteAIDL.java

AIDL的简单使用之通过 IPC 传递对象

如果要跨进程传递某个类,可以通过 IPC 接口来实现。 不过,请务必确保在 IPC 通道的对端可以识别该类的代码,该类必须支持 Parcelable 接口。支持 Parcelable 接口非常重要,因为这使得 Android 系统可将对象分解为能够跨进程组装的原生数据。

可按以下步骤创建支持 Parcelable 协议的类:

  1. 必须实现Parcelable 接口。
  2. 实现 writeToParcel 方法,参数为当前对象的状态,并写入一个 Parcel中。
  3. 在类中添加一个名为 CREATOR 的静态成员变量,即为一个实现了 Parcelable.Creator 接口的对象。
  4. 最后,创建 .aidl 文件,声明该 Parcelable 类(如下述 Rect.aidl 文件所示)。

如果采用自定义编译方式,请不要.aidl 文件加入编译项目。与 C 语言的头文件类似, .aidl 文件不会被编译。AIDL 利用上述方法和成员变量来分解和组装对象。

我们在上述的项目基础上做修改,服务器端:

一、 修改的IRemoteAIDL.aidl文件:

// IRemoteAIDL.aidl
package com.huhx.linux;

import com.huhx.linux.Person;

interface IRemoteAIDL {
    int add(int a, int b);

    List<Person> addPerson(in Person person);
}

二、 增加Person.aidl文件:

// Person.aidl
package com.huhx.linux;

parcelable Person;

三、 增加Person.java文件,注意它的位置是com.huhx.linux;

package com.huhx.linux;

import android.os.Parcel;
import android.os.Parcelable;

/**
 * Created by Linux on 2016/5/9.
 */
public class Person implements Parcelable {
    public String username;
    public String password;

    public Person(String username, String password) {
        this.username = username;
        this.password = password;
    }
    protected Person(Parcel in) {
        username = in.readString();
        password = in.readString();
    }
    public static final Creator<Person> CREATOR = new Creator<Person>() {
        @Override
        public Person createFromParcel(Parcel in) {
            return new Person(in);
        }
        @Override
        public Person[] newArray(int size) {
            return new Person[size];
        }
    };
    @Override
    public int describeContents() {
        return 0;
    }
    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(username);
        dest.writeString(password);
    }
}

这里,之前是放在com.huhx.linux.aidlclient下的,但是处理起来总是报错。不知道是否要求aidl的包名与序列化类的包名一致?

三、 修改RemoteService.java文件

public class RemoteService extends Service {
    List<Person> persons = new ArrayList<>();
    private final static String TAG = "RemoteService";

    private IRemoteAIDL.Stub binder = new IRemoteAIDL.Stub() {
        @Override
        public int add(int a, int b) throws RemoteException {
            Log.i(TAG, "a = " + a + ", b = " + b);
            return a + b;
        }
        @Override
        public List<Person> addPerson(Person person) throws RemoteException {
            persons.add(person);
            return persons;
        }
    };
    @Override
    public IBinder onBind(Intent intent) {
        return binder;
    }
}

我们在上述的项目基础上做修改,客户端:

aidl文件以及Person类与服务器端的一样,这里不做说明,在MainActivity中增加测试的方法。

// 添加一个方法
public void addPerson(View view) {
    Person person1 = new Person("Linux", "123456");
    Person person2 = new Person("Huhx", "654321");
    int length = 0;
    try {
        iRemoteAIDL.addPerson(person1);
        length = iRemoteAIDL.addPerson(person2).size();

        Toast.makeText(MainActivity.this, "list size: " + length, Toast.LENGTH_SHORT).show();
    } catch (RemoteException e) {
        e.printStackTrace();
    }
}
时间: 2024-08-27 00:33:03

android基础---->aidl服务的使用的相关文章

Android Service AIDL 远程调用服务 简单音乐播放实例的实现

Android Service是分为两种: 本地服务(Local Service): 同一个apk内被调用 远程服务(Remote Service):被另一个apk调用 远程服务需要借助AIDL来完成. AIDL 是什么 AIDL (Android Interface Definition Language) 是一种IDL 语言,用于生成可以在Android设备上两个进程之间进行进程间通信(interprocess communication, IPC)的代码.如果在一个进程中(例如Activi

android基础部分再学习--AIDL

AIDL与其他IDL语言类似,你需要做一些工作. 它允许你定义客户端与服务端达成一致的程序接口使用进程间通信相互交流. 在ANdroid上面,一个进程不能正常的访问另一个进程的内存. 所以说,他们需要分解他们的对象为操作系统可以理解的基本单位,然后为你把这些对象按次序跨越进程边界 书写这些代码是单调冗长的,所以android使用AIDL为你处理这个问题. 注意:使用AIDL只有在你允许来自不同应用的客户端跨进程通信访问你的service,并且想要在你的service种处理多线程的时候才是必要的.

Android跨进程访问(AIDL服务)

我将AndroidAIDL的学习知识总结一下和大家共享 在Android开发中,AIDL主要是用来跨进程访问. Android系统中的进程之间不能共享内存,因此,需要提供一些机制在不同进程之间进行数据通信,一般是和Service服务组件一起使用来实现. 1.创建调用AIDL服务 建立AIDL服务的步骤: 第一步:在Eclipse的Android工程的Java源文件目录中建立一个扩展名为aidl的文件,改文件的语法类似于Java代码,但稍有不同. 第二步:如果aidl文件的内容是正确的,ADT会在

Android跨进程通信AIDL服务

服务(Service)是android系统中非常重要的组件.Service可以脱离应用程序运行.也就是说,应用程序只起到一个启动Service的作用.一但Service被启动,就算应用程序关闭,Service仍然会在后台运行. android系统中的Service主要有两个作用:后台运行和跨进程通讯.后台运行就不用说了,当Service启动后,就可以在Service对象中 运行相应的业务代码,而这一切用户并不会察觉.而跨进程通讯是这一节的主题.如果想让应用程序可以跨进程通讯,就要使用我们这节讲的

Android基础入门教程——4.2.3 Service精通

Android基础入门教程--4.2.3 Service精通 标签(空格分隔): Android基础入门教程 本节引言: 本节,我们继续来研究Service(服务)组件,本节将会学习下Android中的AIDL跨进程通信的一些 概念,并不深入到源码层次,暂时知道是什么,会用即可!开始本节内容~ 本节对应官方文档:Binder 1.Binder机制初涉 1)IBinder和Binder是什么鬼? 我们来看看官方文档怎么说: 中文翻译: IBinder是远程对象的基本接口,是饿了高性能而设计的轻量级

android基础知识13:AndroidManifest.xml文件解析

1.重要性 AndroidManifest.xml是Android应用程序中最重要的文件之一.它是Android程序的全局配置文件,是每个 android程序中必须的文件.它位于我们开发的应用程序的根目录下,描述了package中的全局数据,包括package中暴露的组件 (activities, services, 等等),以及他们各自的实现类,各种能被处理的数据和启动位置等重要信息. 因此,该文件提供了Android系统所需要的关于该应用程序的必要信息,即在该应用程序的任何代码运行之前系统所

Android 面试题总结之Android 基础(四)

Android 面试题总结之Android 基础Service(四) 在上一章节Android 面试题总结之Android 基础Broadcast Receiver(三) 我们讲了Broadcast Receiver基础知识.本节主要讲解Service相关基础知识,Service也是我们开发过程中经常使用到. 在阅读过程中有任何问题,请及时联系.如需转载请注明 fuchenxuan de Blog 本章系<Android 之美 从0到1 – 高手之路>Android基础Service 总结了A

Android基础入门教程——10.1 TelephonyManager(电话管理器)

Android基础入门教程--10.1 TelephonyManager(电话管理器) 标签(空格分隔): Android基础入门教程 本节引言: 本章节是Android基础入门教程的最后一章,主要讲解是一些零零散散的一些知识点,以及一些遗漏 知识点的补充,这些零散的知识点包括,各种系统服务的使用,比如本节的电话管理器,短信管理器, 振动器,闹钟,壁纸等等,还有传感器之类的东西!乱七八糟什么都有哈!好的,本节我们要学习的 是TelephonyManager,见名知义:用于管理手机通话状态,获取电

Android基础新手教程——4.2.3 Service精通

Android基础新手教程--4.2.3 Service精通 标签(空格分隔): Android基础新手教程 本节引言: 本节,我们继续来研究Service(服务)组件,本节将会学习下Android中的AIDL跨进程通信的一些 概念,并不深入到源代码层次.临时知道是什么.会用就可以.開始本节内容~ 本节相应官方文档:Binder 1.Binder机制初涉 1)IBinder和Binder是什么鬼? 我们来看看官方文档怎么说: 中文翻译: IBinder是远程对象的基本接口,是饿了高性能而设计的轻