Android服务之Service(三)关于AIDL进程间通信

转载:http://www.cnblogs.com/zhangdongzi/archive/2012/01/09/2317197.html

一.基础知识

AIDL的作用

在Android平台,每个应用程序App都运行在自己的进程空间。通常一 个进程不能访问另一个进程的内存空间(一个应用不能访问另一个应用),如果想沟通,需要将对象分解成操作系统可以理解的基本单元,Android提供了AIDL来处理。

AIDL (Android Interface Definition Language) 是一种IDL 语言,用于生成可以在Android设备上两个进程之间进行进程间通信(interprocess communication, IPC)的代码。如果在一个进程中(例如Activity)要调用另一个进程中(例如Service)对象的操作,就可以使用AIDL生成可序列化的参 数。换句比较浅显的话来说,就是我这个App应用的activity,需要调用其他App应用的Service.当然同一App应用的activity 与service也可以在不同进程间,这可以设置Service配置中,android:process=":remote"

AIDL的使用

官方文档特别提醒我们何时使用AIDL是必要的:只有你允许客户端从不同的应用程序为了进程间的通信而去访问你的service,以及想在你的service处理多线程。(太生硬了,不同进程的组件调用吧。)

那么怎么制作AIDL呢?下面步骤

1:创建.aidl文件。新建立个文件并且以.aidl作为后缀保存,在这个文件中编写接口以及方法,这个我们普通java接口声明是一样的,不同的是要显示import 复杂类型,即便复杂类型对象在同一个包内。Java基本数据类型 (int, long, char, boolean等),String和CharSequence,集合接口类型List和Map,不需要import 。

比如:

package com.dongzi;
interface IStockQuoteService{
  double getPrice(String ticker);
}

2:创建好AIDL文件后,刷新下工程,你会发现在gen包下,对应的包下面出现一个与AIDL文件相同的java文件。如:

/*
 * This file is auto-generated.  DO NOT MODIFY.
 * Original file: D:\\mywordspace\\MyPhone\\src\\com\\dongzi\\IStockQuoteService.aidl
 */
package com.dongzi;

public interface IStockQuoteService extends android.os.IInterface {
    /** Local-side IPC implementation stub class. */
    public static abstract class Stub extends android.os.Binder implements com.dongzi.IStockQuoteService {
        private static final java.lang.String DESCRIPTOR = "com.dongzi.IStockQuoteService";

        /** Construct the stub at attach it to the interface. */
        public Stub() {
            this.attachInterface(this, DESCRIPTOR);
        }

        /**
         * Cast an IBinder object into an com.dongzi.IStockQuoteService
         * interface, generating a proxy if needed.
         */
        public static com.dongzi.IStockQuoteService asInterface(android.os.IBinder obj) {
            if ((obj == null)) {
                return null;
            }
            android.os.IInterface iin = (android.os.IInterface) obj.queryLocalInterface(DESCRIPTOR);
            if (((iin != null) && (iin instanceof com.dongzi.IStockQuoteService))) {
                return ((com.dongzi.IStockQuoteService) iin);
            }
            return new com.dongzi.IStockQuoteService.Stub.Proxy(obj);
        }

        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_getPrice: {
                data.enforceInterface(DESCRIPTOR);
                java.lang.String _arg0;
                _arg0 = data.readString();
                double _result = this.getPrice(_arg0);
                reply.writeNoException();
                reply.writeDouble(_result);
                return true;
            }
            }
            return super.onTransact(code, data, reply, flags);
        }

        private static class Proxy implements com.dongzi.IStockQuoteService {
            private android.os.IBinder mRemote;

            Proxy(android.os.IBinder remote) {
                mRemote = remote;
            }

            public android.os.IBinder asBinder() {
                return mRemote;
            }

            public java.lang.String getInterfaceDescriptor() {
                return DESCRIPTOR;
            }

            public double getPrice(java.lang.String ticker) throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                double _result;
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    _data.writeString(ticker);
                    mRemote.transact(Stub.TRANSACTION_getPrice, _data, _reply, 0);
                    _reply.readException();
                    _result = _reply.readDouble();
                }
                finally {
                    _reply.recycle();
                    _data.recycle();
                }
                return _result;
            }
        }

        static final int TRANSACTION_getPrice = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
    }

    public double getPrice(java.lang.String ticker) throws android.os.RemoteException;
}

AIDL工具自动生成了那么多代码,其实我们只需要知道3个就够了。

public static abstract class Stub extends android.os.Binder implements com.dongzi.IStockQuoteService 静态抽象内部类Stub

private static class Proxy implements com.dongzi.IStockQuoteService         AIDL服务代理类

public double getPrice(java.lang.String ticker) throws android.os.RemoteException;     AIDL公布出的接口,就是我们定义的接口方法

3:把AIDL文件存放在其他客户端应用中,我们这个作为服务端。当然我们也可以方便的把这个应用作为客户端以及服务端。其实根本区别就是为了使调用者与被调用者在不同进程中,于是在服务中添加android:process=":remote"即可。省去了再建立应用客户端调用。

4:AIDL只是请定义一个契约,我们这里需要一个服务来提供服务。于是建立服务MyService.

二.实战

既然我们了解一些基础知识后,现在我们开始来代码吧。假设我们需要在一个进程中调用其他应用的服务,这个服务提供一个股票价格查询,或者GPS定位的服务。

并且定义一个类,继承我们AIDL生成的Stub内部类,并实现我们AIDL定义的方法

代码如下:

package com.dongzi;

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

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

    //定义内部类MyServiceImpl继承我们的AIDL文件自动生成的内部类,
    //并且实现我们AIDL文件定义的接口方法
    private class MyServiceImpl extends IStockQuoteService.Stub{

        @Override
        public double getPrice(String ticker) throws RemoteException {
            Log.e(TAG, "getPrice");
            return 10.5;
        }

    }

    @Override
    public IBinder onBind(Intent arg0) {
        //返回AIDL实现
        return new MyServiceImpl();
    }

    @Override
    public void onDestroy(){
        Log.e(TAG, "Release MyService");
        super.onDestroy();
    }
}

我们需要在onBind方法中返回我们的AIDL接口实现对象,以便其他进程调用。

当然了,现在AIDL,以及Service都定义好了,就需要在mainfest.xml中设置

 <service android:name=".MyService"            android:process=":remote"            >            <intent-filter>                <action android:name="com.dongzi.IStockQuoteService"/>            </intent-filter>        </service>

在客户端服务端在同个App中,android:process=":remote",代表在应用程序里,当需要该service时,会自动创建新的进程。而如果是android:process="remote",没有“:”分号的,则创建全局进程,不同的应用程序共享该进程。

那么现在客户端来调用我们的服务了,代码如下:

package com.dongzi;

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

public class MYyActivity extends Activity {
      static final String TAG="MYyActivity";
       @Override
       public void onCreate(Bundle savedInstanceState){
           super.onCreate(savedInstanceState);
           setContentView(R.layout.main);
           Button btnCall=(Button)findViewById(R.id.btnCall);
           if(btnCall!=null)
               btnCall.setOnClickListener(new OnClickListener(){
                @Override
                public void onClick(View v) {
                    //绑定一个服务
                    bindMyService();
                }
               });
       }

       IStockQuoteService iService=null;
       private ServiceConnection  conn=new ServiceConnection(){
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            //返回AIDL接口对象,然后可以调用AIDL方法
            iService=IStockQuoteService.Stub.asInterface(service);
            double value=0.0;
            try {
                  value=iService.getPrice("");
            }
            catch (RemoteException e) {
                Log.e(TAG,"调用出错!");
                e.printStackTrace();
            }
            Log.e(TAG, "返回数值为:"+value);
        }
        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.i(TAG, "释放Service");
        }
       };

       private void bindMyService(){
           // Intent intent=new Intent("com.dongzi.IStockQuoteService");
           Intent intent=new Intent(this,MyService.class);
           startService(intent);

           bindService(intent, conn, Context.BIND_AUTO_CREATE);
       }
}

在按钮点击时候启动service,然后再绑定这个Service.在连接到服务后,我们会发现,调用AIDL中定义的方法成了,打印如下:

项目结构为:

时间: 2024-10-13 08:53:54

Android服务之Service(三)关于AIDL进程间通信的相关文章

Android:远程服务Service(含AIDL &amp; IPC讲解)

前言 Service作为Android四大组件之一,应用非常广泛 本文将介绍Service其中一种常见用法:远程Service 如果你对Service还未了解,建议先阅读我写的另外一篇文章: Android四大组件:Service史上最全面解析 目录 1. 远程服务与本地服务的区别 远程服务与本地服务最大的区别是:远程Service与调用者不在同一个进程里(即远程Service是运行在另外一个进程):而本地服务则是与调用者运行在同一个进程里 二者区别的详细区别如下图: 2. 使用场景 多个应用程

Android 服务类Service 的详细学习

上一篇说到了通知栏Notification,提起通知栏,不得让人想到Service以及BroadcastReceive,作为android的4大组建的2个重要成员,我们没少和它们打交道.它们可以在无形中使我们的软件和网络.数据库.系统等进行交互,之后通过UI(Notification就是一种展示方式)把结果展现在我们面前.可以说,他们是android生命体系里面的神经系统,通过反射条件让身体展现不同的状态.在整个系统中,广播接收器充当着是传输者和监听者的角色,它把系统的一点点变化都反馈上去,之后

Android服务之Service

Android服务之Service android中服务是运行在后台的东西,级别与activity差不多.既然说service是运行在后台的服务,那么它就是不可见的,没有界面的东西.你可以启动一个服务Service来播放音乐,或者记录你地理信息位置的改变,或者启动一个服务来运行并一直监听某种动作. Service和其他组件一样,都是运行在主线程中,因此不能用它来做耗时的请求或者动作.你可以在服务中开一一个线程,在线程中做耗时动作. 那么究竟Service怎么使用呢? 老规矩,先来点基础知识. 一

Android服务之Service(其一)

转载地址:http://www.cnblogs.com/zhangdongzi/archive/2012/01/08/2316711.html android中服务是运行在后台的东西,级别与activity差不多.既然说service是运行在后台的服务,那么它就是不可见的,没有界面的东西.你可以启动一个服务Service来播放音乐,或者记录你地理信息位置的改变,或者启动一个服务来运行并一直监听某种动作. Service和其他组件一样,都是运行在主线程中,因此不能用它来做耗时的请求或者动作.你可以

Android 服务类Service 的具体学习

上一篇说到了通知栏Notification,提起通知栏,不得让人想到Service以及BroadcastReceive,作为android的4大组建的2个重要成员,我们没少和它们打交道.它们能够在无形中使我们的软件和网络.数据库.系统等进行交互,之后通过UI(Notification就是一种展示方式)把结果展如今我们面前.能够说,他们是android生命体系里面的神经系统,通过反射条件让身体展现不同的状态.在整个系统中,广播接收器充当着是传输者和监听者的角色,它把系统的一点点变化都反馈上去,之后

Android服务之Service(四)--ASDL传递复杂数据对象

此实例与前面aidl不同之处在于,传递的数据比较复杂,传递了自定义数据对象,本实例使用到了两个数据对象Person和Pet,其中Person是作为远程调用Service传递的参数,Pet是远程Service返回调用端的数据.像Java的RMI需要将数据对象序列化一样,此数据对象也需要序列化,因此android要求远程Service传递的参数和返回值需要实现Parcelable接口. 实例化Parcelable接口不仅需要实例化接口中的方法,而且要求在实现类中定义一个名为CREATOR,类型为Pa

Android服务类Service详细解析

Service有什么作用? 许多人不明白service是用来干嘛的,其实Service作为Android四大组件之一,可以理解为一个运行在后台的Activity,它适用于处理一些不干扰用户的长时间的后台操作,比如你播放器播放音乐之后跳到其它页面,音乐需要继续播放,那么这个时候就可以将音乐的播放一直运行在后台服务中,需要启动播放的时候就通过Activity去启动服务,再通过服务去调用播放,需要停止的时候就停止服务. 有人可能会问,Thread也可以实现后台运行,为什么不用Thread而使用Serv

android服务之一 Service

Service是Android系统中四大组件之一(Activity.Service.BroadcastReceiver.ContentProvider),它和Activity的级别差不多,区别在于,Activity有界面显示,而Service是在后台运行,它是不可见的,可以和其他组件进行交互. Service服务的启动有两种方式: 1:本地服务context.startService() context.startService() ->onCreate()- >onStart()->S

Android 中的 Service 三种启动方式

1.start Service    不会随着activity finish 而关闭,必须调用 stop方法 每次调用都会调用onstart方法 package com.weidingqiang.customnetroid; import android.app.Service; import android.content.Intent; import android.os.Bundle; import android.os.IBinder; import android.util.Log;