深入Android架构(从线程到AIDL)_22 IPC的Proxy-Stub设计模式04

目录

[5、 谁来写Proxy及Stub类呢?](#5、 谁来写Proxy及Stub类呢?)

如何考虑人的分工

IA接口知识取得的难题

在编程上,有什么技术可以实现这个方法?

范例


5、 谁来写Proxy及Stub类呢?

-- 强龙提供AIDL工具,给地头蛇产出Proxy和Stub类

如何考虑人的分工
  • 由框架开发者来撰写Proxy-Stub类,才能减轻开发者的负担。
  • 框架分为: <天子框架>和<曹操框架>。
  • 因此, 应该由两者(天子或曹操)之一来撰写Proxy-Stub类。
IA接口知识取得的难题
  • 但是,有个难题: IA接口(如下图所示)的内容必须等到<买主>来了才会知道。
  • 在框架开发阶段,买主还没来, IA接口的知识无法取得,又如何定义IA接口呢? 没有IA接口定义,又如何撰写Stub和Proxy类呢?
  • 好办法是:

"强龙(天子或曹操)撰写代码(在先) ;然后地头蛇(App开发者)定义接口(在后)。 "

在编程上,有什么技术可以实现这个方法?

技术之一是: 類別模板(class template)例如,强龙撰写模板:

java 复制代码
template< class T >
class SomeClass
{
    private:
            T data;
    public:
        SomeClass() { }
        void set(T da)
        { data = da; }
};
  • 地头蛇利用模板来生成一个类:

SomeClass<Integer> x;

由于接口(interface)是一种特殊的类(class),所以也可以定义模板如下:

java 复制代码
template<interface I>
    class BinderProxy
    {
        // .........
    };
  • 地头蛇利用模板来生成一个类:
  • 除了模板之外,还有其它编程技术可以实现<强龙写代码,地头蛇定义接口>的方案吗?
  • 答案是:
    程序生成器(program generator)
    例如: Android的aidl.exe

AIDL

  • AIDL的目的是定义Proxy/Stub来封装IBinder接口,以便产生更亲切贴心的新接口。
  • •所以,在应用程序里,可以选择使用IBinder接口,也可以使用AIDL来定义出接口。
  • AIDL的目的是定义Proxy/Stub来封装IBinder接口,以便产生更亲切贴心的新接口。
  • 所以,在应用程序里,可以选择使用IBinder接口,也可以使用AIDL来定义出新接口。由于IBinder接口只提供单一函数(即transact()函数)来进行远距通信,呼叫起来比较不方便。
  • 所以Android提供aidl.exe工具来协助产出Proxy和Stub类别,以化解这个困难。
  • 只要你善于使用开发环境的工具(如Android的aidl.exe软件工具)自动产生Proxy和Stub类别的程序代码;那就很方便
    了。
范例
  • 此范例使用Android-SDK的/tools/里的aidl.exe工具程序,根据接口定义档(如下述的mp3PlayerInterface.aidl)而自动产出Proxy及Stub类别,其结构如下:
  • 藉由开发工具自动产出Proxy及Stub类的代码,再分别转交给ac01和mp3Binder开发者。此范例程序执行时,出现画面如下:
  • 依据UI画面的两项功能: <Play>和< Stop>,以Java定义接口,如下的代码:
java 复制代码
// mp3PlayerInterface.aidl
interface mp3PlayerInterface mp3PlayerInterface{
    void play();
    void stop();
}
  • 使用Android-SDK所含的aidl.exe工具,将上述的mp3PlayerInterface.aidl档翻译成为下述的mp3PlayerInterface.java档案。
java 复制代码
// mp3PlayerInterface.java
/*
* This file is auto-generated. DO NOT MODIFY.
* Original file: mp3PlayerInterface.aidl
*/
// .........
public interface mp3PlayerInterface extends android.os.IInterface
{
    /** Local-side IPC implementation stub class. */
    public static abstract class Stub extends android.os.Binder implements com.misoo.pkgx.mp3PlayerInterface
    {
        // ..........
        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_play:{
                    data.enforceInterface(DESCRIPTOR);
                    this.play();
                    reply.writeNoException();
                    return true;
                }
                case TRANSACTION_stop:{
                    data.enforceInterface(DESCRIPTOR);
                    this.stop();
                    reply.writeNoException();
                    return true;
                }
            }
            return super.onTransact(code, data, reply, flags);
        }

        private static class Proxy implements com.misoo.pkgx.mp3PlayerInterface
        {
            private android.os.IBinder mRemote;
            //.............
            public void play() throws android.os.RemoteException
            {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                try {
                        _data.writeInterfaceToken(DESCRIPTOR);
                        mRemote.transact(Stub.TRANSACTION_play, _data, _reply, 0);
                        _reply.readException();
                }
                finally {
                    _reply.recycle();
                    _data.recycle();
                } 
            }
        }

        public void stop() throws android.os.RemoteException
        {
            android.os.Parcel _data = android.os.Parcel.obtain();
            android.os.Parcel _reply = android.os.Parcel.obtain();
            try {
                _data.writeInterfaceToken(DESCRIPTOR);
                mRemote.transact(Stub.TRANSACTION_stop, _data, _reply, 0);
                _reply.readException();
            }
            finally {
                _reply.recycle();
                _data.recycle();
            }
        }

        static final int TRANSACTION_play = (IBinder.FIRST_CALL_TRANSACTION + 0);
        static final int TRANSACTION_stop = (IBinder.FIRST_CALL_TRANSACTION + 1);
    }
    public void play() throws android.os.RemoteException;
    public void stop() throws android.os.RemoteException;
}
  • 表面上,此mp3PlayerInterface.java是蛮复杂的,其实它的结构是清晰又简单的,只要对于类继承、反向調用和接口等面向对象观念有足够的认识,就很容易理解了。
java 复制代码
// mp3Binder.java
package com.misoo.pkgx;
import android.content.Context;
import android.media.MediaPlayer;
import android.util.Log;

public class mp3Binder extends mp3PlayerInterface.Stub{
    private MediaPlayer mPlayer = null;
    private Context ctx;
    public mp3Binder(Context cx){ ctx= cx; }
    
    public void play(){
        if(mPlayer != null) return;
        mPlayer = MediaPlayer.create(ctx, R.raw.test_cbr);
        try { mPlayer.start();
        } catch (Exception e)
        { Log.e("StartPlay", "error: " + e.getMessage(), e); }
    }

    public void stop(){
        if (mPlayer != null)
            { mPlayer.stop(); mPlayer.release(); mPlayer = null; }
    }
}
  • 撰写mp3RemoteService类
java 复制代码
// mp3Service.java
package com.misoo.pkgx;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;

public class mp3Service extends Service {
    IBinder ib = null;
    @Override public void onCreate() {
        super.onCreate();
        ib = new mp3Binder(this.getApplicationContext());
    }

    @Override public void onDestroy() { }
    @Override public IBinder onBind(Intent intent) {return ib;}
}

// ac01.java
// .........
public class ac01 extends Activity implements OnClickListener {
    //..........
    private PlayerProxy pProxy = null;
    public void onCreate(Bundle icicle) {
        // .........
        startService(new Intent("com.misoo.pkgx.REMOTE_SERVICE"));
        bindService(new Intent("com.misoo.pkgx.REMOTE_SERVICE"),mConnection, Context.BIND_AUTO_CREATE);
    }

    private ServiceConnection mConnection = new ServiceConnection() {
        public void onServiceConnected(ComponentName className,IBinder ibinder) {
            pProxy = mp3PlayerInterface.Stub.asInterface(ibinder);
        }
        public void onServiceDisconnected(ComponentName className) {}
    };

    public void onClick(View v) {
        switch (v.getId()) {
            case 101: pProxy.play(); tv.setText(pProxy.getStatus()); break;
            case 102: pProxy.stop(); tv.setText(pProxy.getStatus()); break;
            case 103:
                unbindService(mConnection);
                stopService(new Intent(
                    "com.misoo.pkgx.REMOTE_SERVICE"));
                    finish(); break;
        }
    }
}
  • 对于Anrdoid的初学者而言, Android的 AIDL机制可说是最难弄懂的。
相关推荐
思忖小下1 个月前
深入Android架构(从线程到AIDL)_19 IPC的Proxy-Stub设计模式01
ipc·proxy-stub
思忖小下1 个月前
深入Android架构(从线程到AIDL)_20 IPC的Proxy-Stub设计模式02
ipc·proxy-stub
微澜-3 个月前
海康IPC接入TRTC时,从海康中获取的数据显示时色差不正确
音视频·ipc·trtc
小乖兽技术3 个月前
C#与C++交互开发系列(二十):跨进程通信之共享内存(Shared Memory)
c++·c#·交互·ipc
小乖兽技术3 个月前
C#与C++交互开发系列(十九):跨进程通信之套接字(Sockets)
c++·c#·交互·ipc
Hdnw3 个月前
Android IPC机制(一)多进程模式
android·ipc
bbqz0076 个月前
逆向WeChat (五)
c++·微信·逆向·mojo·ipc·wechat·mmmojo
林多9 个月前
【通信中间件】Fdbus HelloWorld实例
c++·rpc·ipc·例子·fdbus·通信中间件
yangwei0251 年前
【案例】IPC 中的WinCC RT Advanced PC项目,如何下载及开机自动启动?
ipc·wincc rt adv 应用