AIDL

2016-11-01  本文已影响0人  A_Coder

AIDL(Android接口描述语言)是一种接口描述语言,通常用于进程间通信。编译器根据AIDL文件生成一个系列对应的Java类,通过预先定义的接口以及Binder机制达到进程间通信的目的。说白了,AIDL就是定义一个接口,客户端(调用端)通过bindService()来与远程服务端建立一个连接,在该连接建立时会返回一个IBinder对象,该对象是服务端Binder的BinderProxy,在建立连接时,客户端通过asInterface函数将该BinderProxy对象包装本地的Proxy,并将远程服务端的BinderProxy对象赋值给Proxy类的mRemote字段,就是通过mRemote执行远程函数调用。
在客户端新建一个AIDL文件,在SsoAuth.aidl文件中会默认有一个basicTypes函数,我们在程序后面添加一个ssoAuth的函数用于SSO授权。

interface SsoAuth{
    void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString);
     //实现SSO授权
     void ssoAuth(String username, String pwd);
}

因为客户端是调用端,因此,只需要定义AIDL文件,此时Rebuild一下工厂就会生成一个SsoAuth.java类,该类根据SsoAuth.aidl文件生成,包含了我们在AIDL文件中定义的函数。因为AIDL通常用于进程间通信,因此,我们新建一个被调用端的工程,命名为aidl_service,然后将客户端的AIDL文件夹复制到aidll_service的app/src/main目录下。
此时相当于在客户端和被调用端都有同一份SsoAuth.aidl文件,它们的包名和类名完全一致,生成的SsoAuth.java类也完全一致,这样在远程调用时,它们就能够拥有一致的类型。Rebuild被调用端工程之后就会生成SsoAuth.java文件,该文件中有一个Stub的子类,并且在Service的onBind()函数中返回这个Stub子类的对象。

public class SinaSsoAuthService extends Service{
    SinaSsoImpl mBinder = new SinaSsoImpl();
    
    @Override
    public void onCreate(){
        super.onCreate();
        Log.e("","## sso auth create");
    }
    @Nullable
    @Override
    public IBinder onBind(Intent intent){
        return mBinder;
    }
    //继承自Stub类,在这里实现ssoAuth函数
    class SinaSsoImpl extentds SsoAuth.Stub{
        @Override
        public void ssoAuth(String userName, String pwd) throws RemoteException{
            Log.e("", "这里是新浪客户端,执行SSO登录,用户名:" + username + ",密码:" + pwd);
        }

        @Override
        void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException{
        }
    }
    
}

实际上完成功能的是继承Stub的SinaSsoImpl类,Service只提供了一个让SinaSsoImpl依附的外壳。需要把它注册在AndroidManifest.xml中:

<service
    andorid:name=".service.SinaSsoAuthService"
    android:exported="true"  //表示该服务是否能够被其他应用程序所控制或连接
    android:process=":remote" //表示该服务是否运行在另外一个进程,如果设置了此项,那么在包名后面加上这段字符串表示另一进程的名字
    android:label="@string/app_name" //服务的名字
    >
    <inter-filter>
        <action android:name="book.aidl_service.service.SinaSsoAuthService" />
    </inter-filter>
</service>

然后先运行被调用端(也就是Service端)应用,并且在客户端中完成调用Service的代码。客户端Activity的代码如下:

public class MainActivity extends AppCompatActivity{
    SsoAuth mSsoAuth;
    @Override
    protected void onCreate(Bundle saveinstanceState) {
        super.onCreate(saveInstanceState);
        setContentView(R.layout.activity_main);
        //执行操作
        findViewById(R.id.sso_th).setOnClickListener(new View.OnClickListner(){
            @Override
            public void onClick(View view){
                //绑定远程服务,并且进行登陆
                bindSsoAuthService();
            } else
                  doSsoAuth();
        });
    }
    public void bindSsoAuthService(){
        Intent intent = new Intent("book.aidl_server.service.SinaSsoAuthService");
        bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
    }

    ServiceConnection mConnection = new ServiceConnection(){
        @Override
        public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
            //建立连接后将Binder转换为mSsoAuth
            mSsoAuth = SsoAuth.Stub.asInterface(iBinder);
            doSsoAuth();
        }
        @Override
        public void onServiceDisconnected(ComponentName componentName){
            mSsoAuth = null;
        }
    };

    private void doSsoAuth(){
        try{
            //执行登录,实际上调用的是Server端的ssoAuth函数
            mSsoAuth.ssoAuth("coder", "pwd123");
        }catch(RemoteException e){
            e.printStackTrace();
        }
    }
    @Override
    protected void onDestroy(){
        super.onDestroy();
        unbindService(mConnection);
    }
}

在上述Activity程序中,运行程序后点击登录按钮时会向Server端发起连接Service请求,在建立连接后会将Binder对象转换为SsoAuth对象,然后调用ssoAuth函数。此时的ssoAuth函数实际上调用的就是Server端中SinaSsoImpl类的实现。
这一切的核心都是通过AIDL文件生成的Stub类以及其背后的Binder机制。


注:
以上是Android开发进阶从小工到专家笔记

上一篇下一篇

猜你喜欢

热点阅读