一、AIDL Service简介
Android系统中,各个应用都运行在自己的进程中,进程之间一般无法直接进行通信,为了实现进程通信,Android提供了AIDL Service;
二、与本地Service不同
本地Service:直接把IBinder对象本身传递给客户端的ServiceConnection的onServiceConnected方法的第二个参数;远程Service:只将IBinder对象的代理传给客户端的ServiceConnection的onServiceConnected方法的第二个参数;
三、AIDL文件
Android需要AIDL(Android Interface Definition Language)来定义远程接口,这种接口定义语言并不是一种真正的变成语言,只是定义两个进程之间的通信接口;
与Java接口相似,但是存在如下几点差异:
AIDL定义接口的源代码必须以.aidl结尾;
AIDL用到的数据类型,除了基本类型、String、List、Map、CharSequence之外,其它类型全部都需要导包,即使它们在同一个包中也需要导包;
四、例子:
1. 创建AIDL文件,定义好的AIDL文件后,ADT工具会自动在gen目录下生成一个AIDL.java接口,该类内部包含一个Stub内部类,实现了IBinder,AIDL里面的接口,这个Stub类会作为远程Service回调类:
IMyService.aidl
package com.juno.serviceaidltest; import com.juno.serviceaidltest.Product;
interface IMyService
{ String getValue(); Map getMap(in String country, in Product product); Product getProduct();
}
Product.aidl
parcelable Product;
Product.java
package com.juno.serviceaidltest; import android.os.Parcel;
import android.os.Parcelable; public class Product implements Parcelable { private int id; private String name; private float price; public static final Parcelable.Creator<Product> CREATOR = new Parcelable.Creator<Product>() { public Product createFromParcel(Parcel in) { return new Product(in); } public Product[] newArray(int size) { return new Product[size]; } }; public Product() { } private Product(Parcel in) { readFromParcel(in); } @Override public int describeContents() { return 0; } public void readFromParcel(Parcel in) { id = in.readInt(); name = in.readString(); price = in.readFloat(); } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeInt(id); dest.writeString(name); dest.writeFloat(price); } public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public float getPrice() { return price; } public void setPrice(float price) { this.price = price; } }
2. 将接口暴露给客户端
MyService.java
package com.juno.serviceaidltest; import java.util.HashMap;
import java.util.Map; import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException; public class MyService extends Service { /** * 继承Stub,也就是实现了IMyService接口,并实现了IBinder接口 */ public class MyServiceImpl extends IMyService.Stub { @Override public String getValue() throws RemoteException { return "Test Value"; } @Override public Map<String, Object> getMap(String country, Product product) throws RemoteException { Map<String, Object> map = new HashMap<String, Object>(); map.put("country", country); map.put("id", product.getId()); map.put("name", product.getName()); map.put("price", product.getPrice()); map.put("product", product); return map; } @Override public Product getProduct() throws RemoteException { Product product = new Product(); product.setId(1234); product.setName("汽车"); product.setPrice(31000); return product; } } @Override public IBinder onBind(Intent intent) { /** * 返回MyServiceImpl对象,在绑定本地Service情况下,该MyServiceImpl会直接传给客户端的ServiceConnected对象的ServiceConnected()方法的第二个参数;在绑定远程Service的情况下,只将MyServiceImpl对象的代理传给客户端的ServiceConnected对象的ServiceConnected()方法的第二个参数 */ return new MyServiceImpl(); } }
3. 在AndroidManifext.xml文件中配置该Service:
<service android:name=".MyService" > <intent-filter> <action android:name="com.juno.serviceaidltest.IService" /> </intent-filter> </service>
4. 在Activity里访问 AIDLService,如果不在同一个App下面访问,需要将Service端的AIDL文件复制到客户端中,并在相同的包名下:
MainActivity.java
package com.juno.serviceanotheraidltest; 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.widget.Button;
import android.widget.TextView; import com.juno.serviceaidltest.IMyService; public class MainActivity extends Activity implements View.OnClickListener { private final static String ACTION = "com.juno.serviceaidltest.IService"; private IMyService myService = null; private Button mBtnInvokeAIDLService; private Button mBtnBindAIDLService; private TextView mTextView; private ServiceConnection mServiceConnection = new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) { //获取远程Service的onBinder方法返回的对象代理 myService = IMyService.Stub.asInterface(service); mBtnInvokeAIDLService.setEnabled(true); try { Log.v("juno", myService.getValue()); } catch (RemoteException e) { e.printStackTrace(); } } @Override public void onServiceDisconnected(ComponentName name) { myService = null; } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initView(); } private void initView() { mBtnInvokeAIDLService = (Button) findViewById(R.id.btnInvokeAIDLService); mBtnBindAIDLService = (Button) findViewById(R.id.btnBindAIDLService); mBtnInvokeAIDLService.setEnabled(false); mTextView = (TextView) findViewById(R.id.textView1); mBtnInvokeAIDLService.setOnClickListener(this); mBtnBindAIDLService.setOnClickListener(this); } @Override public void onClick(View view) { switch (view.getId()) { case R.id.btnBindAIDLService: //创建所需要绑定的Service的Intent,绑定远程的服务 bindService(new Intent(ACTION), mServiceConnection, Context.BIND_AUTO_CREATE); break; case R.id.btnInvokeAIDLService: try { String s = myService.getValue(); s = "Product.id = " + myService.getProduct().getId() + "\n"; s += "Product.name = " + myService.getProduct().getName() + "\n"; s += "Product.price = " + myService.getProduct().getPrice() + "\n"; s += myService.getMap("China", myService.getProduct()).toString(); mTextView.setText(myService.asBinder().isBinderAlive() + " " + s); } catch (Exception e) { } break; } } @Override protected void onDestroy() { super.onDestroy(); if (myService != null) { //解除绑定 unbindService(mServiceConnection); } } }
布局文件
activity_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" > <TextView android:id="@+id/textView1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/hello_world" /> <Button android:id="@+id/btnBindAIDLService" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_below="@id/textView1" android:text="btnBindAIDLService" /> <Button android:id="@+id/btnInvokeAIDLService" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_below="@id/btnBindAIDLService" android:text="btnInvokeAIDLService" /> </RelativeLayout>