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开发进阶从小工到专家笔记