Binder是Android特有的进程间通信(IPC)机制和远程方法调用系统,整个核心平台的跨进程操作几乎都是通过binder进行的,相对于其他的IPC方式,binder有如下特点:
- 通过驱动程序来进行进程间通信
- 通过共享内存来提高性能
- 每个进程都有处理请求的线程池
- 引用计数功能,支持跨进程的对象引用
- 进程间同步调用
- 可以跨进程传输文件描述符
- 服务端终止通知
这篇文章我通过一个实际的例子来演示如何创建一个本地的binder服务,并验证几个binder的特性,主要演示以下几个功能:
1. 客户端通过binder获取一块共享的匿名内存,把内容给打印出来
2. 客户端通过binder获取一个服务端已经打开的文件的文件描述符,写入内容,并在服务端打印出文件的内容
3. 客户端通过binder设置和获取服务端的一个整数成员
4. 客户端通过binder设置 和获取服务端一个字符串成员
5. 验证binder的death notification功能
6. 通过dumpsys工具获取服务端的信息
演示程序已经上传到github,通过下面的命令获取:
git clone https://github.com/mason-Wang/binderipctest.git
1.定义接口
//IIpcTestService.h
class IIpcTestService : public IInterface
{
public:
DECLARE_META_INTERFACE(IpcTestService);
virtual sp<IMemory> getMemory() const = 0;
virtual int getFileDescriptor() = 0;
virtual void dumpFile() = 0;
virtual void setInt(int value) = 0;
virtual int getInt() = 0;
virtual void setString(const char* str) = 0;
virtual char* getString() = 0;
};
class BnIpcTestService : public BnInterface<IIpcTestService>
{
public:
virtual status_t onTransact(uint32_t code,
const Parcel& data,
Parcel* reply,
uint32_t flags = 0);
};
这个接口定义了客户端和服务端进程间通信所用到的所有接口,客户端和服务端都会继承这个接口,并实现。BnIpcTestService
是服务端的代理。
2.实现接口
// IIpcTestService.cpp
enum {
GET_MEMORY = IBinder::FIRST_CALL_TRANSACTION,
GET_FD,
DUMP_FILE,
SET_INT,
GET_INT,
SET_STR,
GET_STR,
};
class BpIpcTestService : public BpInterface<IIpcTestService>
{
public:
BpIpcTestService(const sp<IBinder>& impl)
: BpInterface<IIpcTestService>(impl)
{
}
virtual sp<IMemory> getMemory() const
{
Parcel data, reply;
sp<IMemory> mem;
data.writeInterfaceToken(IIpcTestService::getInterfaceDescriptor());
status_t status = remote()->transact(GET_MEMORY, data, &reply);
if (status == NO_ERROR) {
mem = interface_cast<IMemory>(reply.readStrongBinder());
if (mem != 0 && mem->pointer() == NULL) {
mem.clear();
}
}
return mem;
}
virtual int getFileDescriptor()
{
Parcel data, reply;
int fd;
data.writeInterfaceToken(IIpcTestService::getInterfaceDescriptor());
status_t status = remote()->transact(GET_FD, data, &reply);
if (status == NO_ERROR