简书 android binder,Android Binder的极简使用

进程间通信很多同学都使用到AIDL,这个是对Binder进行了一层封装。其实剥开AIDL,刺果果的使用Binder,有种很简单的方式,不过最好是系统应用,因为看Android版本的提升,在安全方面一直在完善,不排除以后只能系统权限才能使用这种方式。好了,Read the code~

服务端,首先要有一个Binder类,然后重写onTransact

public class MyBinder extends Binder {

String TAG = "MyBinder";

private Context context = null;

public MyBinder(Context context) {

// TODO Auto-generated constructor stub

this.context = context;

}

@Override

protected boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException {

switch (code) {

case 0://

//int len = data.readInt();

reply.writeInt(1);

Log.d(TAG, "onTransact case 0");

return true;

case 1://

return true;

}

return super.onTransact(code, data, reply, flags);

}

}

接着,搞一个服务,通过反射调用系统方法将其添加到servicemanager,早期Android版本貌似有直接的API,现在都需要反射调用了。

IBinder mb = new MyBinder(context) ;

try {

Class> serviceManager = Class.forName("android.os.ServiceManager");

Method method=serviceManager.getMethod("addService", String.class, IBinder.class);

method.invoke(null, "testService", mb);

Log.d(TAG, "add testService to systemservice");

} catch (Exception e) {

// TODO Auto-generated catch block

Log.i(TAG, "add testService fail");

e.printStackTrace();

}

服务端差不多就这样了,看看客户端怎么和这个玩意儿通信

先获取服务,

try {

Class> serviceManager = Class.forName("android.os.ServiceManager");

Method method=serviceManager.getMethod("getService", String.class);

ibinder = (IBinder) method.invoke(null, "testService");

Log.i("wwwwwww", "get testService success ibinder:"+ibinder);

} catch (Exception e) {

// TODO: handle exception

Log.i("wwwwwww", "get testService fail");

}

接着,这样使用

Parcel data =Parcel.obtain();

Parcel reply=Parcel.obtain();

byte[] cmdsetb = {0x02,0x02,0x21,0x44,0x02,0x14,0x09,0x42,0x08,0x00,0x00,0x20};

data.writeInt(12);

data.writeByteArray(cmdsetb);

try {

boolean c = ibinder.transact(0, data, reply, 0);

int ir = reply.readInt();

} catch (RemoteException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

这就把数据(data)发过去了,然后服务端可以通过replay返回数据;第一个参数是code;最后一个是flags,表示同步或异步。

是不是比封装AIDL简单。了解AIDL的同学看到上面的代码可能会注意到一点,就是服务端返回数据,即可对应到AIDL的out回传方式。差别在于AIDL是回传引用,所以像上面代码直接往replay写int那样的方式AIDL就做不到了。AIDL的话可以new一个类,传引用回去。

============================================================

客户端和服务端绑定后,可以监听服务端的状态,当服务端因为异常停止后,能收到死亡通知。如下

直接用binder的linkToDeath

try {

ibinder.linkToDeath(deathHandle, 0);

} catch (RemoteException e2) {

// TODO Auto-generated catch block

e2.printStackTrace();

}

final DeathRecipient deathHandle = new DeathRecipient(){

@Override

public void binderDied() {

// TODO Auto-generated method stub

Log.i("wwss", "binder is died");

}

};

这样,当服务端崩溃的时候,binder断开,即可接收到死亡通知。

如果是服务端需要监听客户端是否崩溃、被kill呢,又该如何?

那就有点不同了,因为客户端是不确定的,所以需要客户端注册一个binder进来。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值