要想了解Binder的原理,结合使用方便我们更简单的去理解调用逻辑。AIDL :Android Interface definition language 我们可以理解为Android 接口定义语言。通过它可以实现进程间的通讯(IPC),如何实现AIDL呢? 进程间通讯需要两个进场,体现在Android中就需要两个App(一个作为Server,一个作为Client).接下来看如何使用AIDL来进行进程间通讯.
Server
- 1.新建工程:包名 com.jfson.aidlserver
- 2.新建model类,实现Parcelable序列化,便于后续在AIDL中进行传递。
public class User implements Parcelable {
public String name;
public int id;
public User(String name, int id) {
this.name = name;
this.id = id;
}
/**
* 序列化
*/
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(name);
dest.writeInt(id);
}
@Override
public int describeContents() {
// 大都返回0即可
return 0;
}
public static final Creator<User> CREATOR = new Creator<User>() {
@Override
public User createFromParcel(Parcel in) {
return new User(in);
}
@Override
public User[] newArray(int size) {
return new User[size];
}
};
/**
* 反序列化得到的 Parcel 构造对象
*/
protected User(Parcel parcel) {
name = parcel.readString();
id = parcel.readInt();
}
}
- 3.新建AIDL文件User.aidl,并引入User类
// User.aidl
package com.jfson.aidlserver;
parcelable User;
- 4.在包名目录下新建AIDL文件IMyAidlService.aidl,其实是个接口,主动实现了一个方法 void basicTypes()。在添加一个我们需要的方法 String giveMessage()。
- 注意:这里需要导入User类的包名。
- 方法参数中,除了基本数据类型外都需要标上类型 :in(输入), out(输出), inout(输入输出)
package com.jfson.aidlserver;
// Declare any non-default types here with import statements
import com.jfson.aidlserver.User;
interface IMyAidlService {
/**
* Demonstrates some basic types that you can use as parameters
* and return values in AIDL.
*/
void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
double aDouble, String aString);
User giveMessage(in User user);
}
- 3.新建 Service 并在OnBinder中返回 Aidl接口的实现类。在实现giveMessage方法中,返回你想提供给Client app的信息即可。
public class MyService extends Service {
public MyService() {
}
@Override
public IBinder onBind(Intent intent) {
// TODO: Return the communication channel to the service.
// 返回 service接口的实例
return new MyServiceImpl();
}
public class MyServiceImpl extends IMyAidlService.Stub {
@Override
public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException {
}
@Override
public User giveMessage(User user) throws RemoteException {
if (user != null){
user.name += " is niubility!";
}
return user;
}
}
}
- 3.1 清单文件注册Service
<service
android:name="com.jfson.aidlserver.MyService"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.jfson.aidlserver.IMyAidlService"/>
</intent-filter>
</service>
</application>
- 4.到此时,作为Server的App 已经编写完毕,编译后会生成java代码。目录在:app/build/generated/source/aidl/debug/com.jfson.aidlserver/IMyAidlService。
生成的代码需要提供给Client App使用的。稍后再分析生成的代码。
Client
1.新建工程: 包名 com.jfson.aidltest
2.将Server端自动生成AIDL的java代码和User.java拷过来,并保持目录一致(com.jfson.aidlserver/IMyAidlService)。
3.绑定服务
public class MainActivity extends AppCompatActivity {
private Button button;
private IMyAidlService myService;
private ServiceConnection mConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// 获取服务对象
myService = IMyAidlService.Stub.asInterface(service);
Toast.makeText(getBaseContext(), "onServiceConnected", Toast.LENGTH_LONG).show();
}
@Override
public void onServiceDisconnected(ComponentName name) {
myService = null;
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button = (Button) findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
try {
if (myService == null) {
// onServiceConnected 回调完成后才能得到接口
return;
}
User user = myService.giveMessage(new User("Android", 110));
if (user != null) {
Toast.makeText(getBaseContext(), user.name, Toast.LENGTH_LONG).show();
}
} catch (RemoteException e) {
e.printStackTrace();
}
}
});
Intent intent = new Intent();
// 目标Service action
intent.setAction("com.jfson.aidlserver.IMyAidlService");
// 5.0以上,需要指定包名
intent.setPackage("com.jfson.aidlserver");// the service package
// 绑定服务,可设置或触发一些特定的事件
bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
}
}