为什么android要使用Binder机制
1.linux中大多数标准 IPC 场景(如管道、消息队列、ioctl 等)的进程间通信机制
+------------------+ +------------------+ +------------------+
| 用户进程 A | | 内核空间 | | 用户进程 B |
| (User Space A) | | (Kernel Space) | | (User Space B) |
+------------------+ +------------------+ +------------------+| | || 1. copy_from_user() | ||----------------------------> || | || | || | 2. copy_to_user() || | -------------------------->|| | |
📝 图解说明:
-
用户进程 A 将数据从用户空间拷贝到内核空间:
- 使用
copy_from_user()
函数。 - 此操作将数据从用户缓冲区复制到内核缓冲区。
- 使用
-
内核保存数据:
- 数据暂时存储在内核中的某个缓冲区(如驱动私有缓冲区、socket 缓冲区等)。
-
用户进程 B 从内核空间读取数据:
- 使用
copy_to_user()
函数。 - 此操作将数据从内核缓冲区复制到用户进程 B 的缓冲区。
- 使用
✅ 总结:
- 发生了 单向通信两次数据复制(物理内存拷贝)(用户 → 内核 → 用户)。
- 虽然性能不如共享内存等零拷贝机制,但适用于大多数标准 IPC 场景(如管道、消息队列、ioctl 等)。
缺点: 物理内存拷贝(Physical Memory Copy)是指在不同内存区域之间实际复制数据内容,而不是通过指针映射或页表共享来访问同一块内存。它对系统性能、资源利用率和程序响应性都有直接影响。
🧠 物理内存拷贝会影响哪些方面?
影响维度 | 描述 |
---|---|
性能 | 拷贝大量数据会消耗 CPU 时间,影响整体执行效率。 |
延迟 | 数据拷贝耗时会导致操作延迟,特别是在实时系统中影响用户体验。 |
吞吐量 | 频繁的内存拷贝会降低系统的并发处理能力。 |
内存带宽 | 大量拷贝占用内存总线带宽,可能成为瓶颈。 |
能耗 | 在移动设备上,频繁拷贝会增加功耗,影响电池寿命。 |
物理内存拷贝会影响性能、延迟、内存带宽和能耗,尤其是在大数据传输或高频率调用场景下尤为明显。
推荐做法:
- 小数据量通信:可以接受一定拷贝开销,使用标准 IPC(如 Binder、Pipe)即可。
- 大数据量通信:优先考虑 共享内存、mmap、MemoryFile、ashmem 等零拷贝方案。
- 实时性要求高:避免使用同步阻塞 IPC,推荐异步 + 零拷贝机制。
2.Binder机制
不同进程通信时,只有在客户端拿到代理类调用方法的时候,才是Binder进行用户空间和内核空间数据传输的过程!!!
步骤 | 操作描述 | 技术细节(以数据发送为例) |
---|---|---|
步骤 1 | Binder 驱动准备 | 为进程通信做准备,实现“内存映射(mmap)”,创建接收缓存区、映射关系(内核缓存区 ↔ Server 用户空间) |
步骤 2 | Client 发送数据 | Client 进程通过系统调用 copy_from_user ,将数据从“用户空间”拷贝到“内核缓存区”,再由 Binder 驱动通知 Server 进程 |
步骤 3 | Server 处理请求 | Server 进程被 Binder 驱动唤醒后,从内核缓存区读取数据、解析,调用目标方法处理 |
步骤 4 | Server 返回结果 | Server 进程通过 copy_to_user ,将处理结果从内核缓存区拷贝到 Client 进程的用户空间,Binder 驱动通知 Client 进程 |
三、优点总结
- 传输效率高:数据拷贝次数少(仅 1 - 2 次 ),依托内存映射让“用户空间 ↔ 内核空间”直接共享数据,减少冗余操作。
- 灵活适配:为接收进程动态分配“接收缓存区”,无需提前固定大小,适配不同数据量的通信场景。
(客户端发送数据:客户端通过 transact() 方法发送数据时,只需将数据写入 Parcel 并传递给 Binder 驱动,无需提前声明数据长度。
服务端接收缓存区创建:Binder 驱动根据客户端实际发送的数据量,为服务端进程动态分配足够大小的内核缓存区(通过 mmap 映射到用户空间),用于接收数据。 ) - 死亡通知机制
// 客户端注册死亡通知
mRemote.linkToDeath(new DeathRecipient() {@Overridepublic void binderDied() {// 服务进程崩溃后的回调}
});
实现原理:驱动检测到服务进程退出时,发送 BR_DEAD_BINDER 消息给所有客户端。
3.代码举例
代码中的1-4是代码执行流程!!!!
以下是一个完整的 AIDL 跨进程通信示例,包含服务端和客户端的 Java 代码,实现简单的加法运算功能:
一、服务端代码
1. 创建 AIDL 文件
在 Android 项目的 main
目录下,新建 aidl
文件夹(与 java
目录同级 ),然后创建包名对应的目录(如 com/example/aidlserver
),在该目录下新建 IMyAidlInterface.aidl
文件:
// IMyAidlInterface.aidl
package com.example.aidlserver;interface IMyAidlInterface {int add(int num1, int num2);
}
2. 编写 Service 类
创建 IRemoteService.java
,继承 Service
,实现 AIDL 接口的 Stub
:
package com.example.aidlserver;import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;public class IRemoteService extends Service {private static final String TAG = "IRemoteService";2.实现 AIDL生成的Stub 类(Stub是服务端Binder实现)private IBinder iBinder = new IMyAidlInterface.Stub() {@Overridepublic int add(int num1, int num2) throws RemoteException {return num1 + num2;}};3.接受客户端发来的bind请求,返回服务端Binder@Overridepublic IBinder onBind(Intent intent) {return iBinder;}@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {Log.d(TAG, "服务已启动");return super.onStartCommand(intent, flags, startId);}
}
3. 注册 Service 到清单文件
在 AndroidManifest.xml
中声明 Service
,设置可被其他进程访问:
<serviceandroid:name=".IRemoteService"android:exported="true"><intent-filter><action android:name="com.example.aidlserver.IRemoteService" /></intent-filter>
</service>
二、客户端代码
1. 复制 AIDL 文件
将服务端的 IMyAidlInterface.aidl
文件,完整复制到客户端项目的 aidl
目录(同样保持包名路径 com/example/aidlserver
)。
2. 编写 Activity 类
创建 MainActivity.java
,绑定服务端 Service
并调用 AIDL 方法:
package com.example.aidlclient;import android.content.ComponentName;
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 androidx.appcompat.app.AppCompatActivity;import com.example.aidlserver.IMyAidlInterface;public class MainActivity extends AppCompatActivity {private static final String TAG = "MainActivity";private IMyAidlInterface aidlInterface;private TextView resultTv;private IBinder mRemote; // 保存服务端 Binder 引用private ServiceConnection serviceConnection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {4.服务绑定后,根据传过来的service获取服务端Binder的代理对象!!!// 将 IBinder 转换为 AIDL 接口aidlInterface = IMyAidlInterface.Stub.asInterface(service);// 2. 保存 Binder 引用,用于注册死亡通知mRemote = service;// 3. 注册 Binder 死亡通知(关键步骤)try {mRemote.linkToDeath(deathRecipient, 0);} catch (RemoteException e) {e.printStackTrace();}Log.d(TAG, "服务已连接");}@Overridepublic void onServiceDisconnected(ComponentName name) {aidlInterface = null;Log.d(TAG, "服务已断开");}};// 死亡通知回调实现private IBinder.DeathRecipient deathRecipient = new IBinder.DeathRecipient() {@Overridepublic void binderDied() {// 服务端进程崩溃时的处理逻辑runOnUiThread(() -> {Toast.makeText(MainActivity.this, "服务已崩溃", Toast.LENGTH_SHORT).show();// 重置引用,可选择重新绑定服务if (mRemote != null) {mRemote.unlinkToDeath(this, 0);mRemote = null;}aidlInterface = null;// 可添加自动重连逻辑bindService(...);});}};性能开销:频繁注册 / 解除死亡通知会有一定开销,建议在应用生命周期内合理管理。public void binderDied() {// 重新绑定服务bindService(new Intent(MainActivity.this, IRemoteService.class),serviceConnection, BIND_AUTO_CREATE);}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);resultTv = findViewById(R.id.result_tv);Button bindBtn = findViewById(R.id.bind_btn);Button callBtn = findViewById(R.id.call_btn);// 绑定服务bindBtn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent();// 设置服务端包名和 Service actionintent.setPackage("com.example.aidlserver");intent.setAction("com.example.aidlserver.IRemoteService");1.绑定服务bindService(intent, serviceConnection, BIND_AUTO_CREATE);}});// 调用 AIDL 方法callBtn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {if (aidlInterface != null) {try {int result = aidlInterface.add(5, 3);resultTv.setText("计算结果:" + result);} catch (RemoteException e) {e.printStackTrace();}} else {resultTv.setText("服务未连接");}}});}@Overrideprotected void onDestroy() {super.onDestroy();// 解绑服务unbindService(serviceConnection);}
}
具体流程拆解
一、流程拆解 + 代码对应(结合之前 AIDL 示例)
1. 客户端 bindService
客户端通过 bindService
发起绑定服务的请求,代码如下(以客户端 MainActivity
为例):
Intent intent = new Intent();
intent.setPackage("com.example.aidlserver"); // 服务端包名
intent.setAction("com.example.aidlserver.IRemoteService"); // 服务端 Service 的 action
bindService(intent, serviceConnection, BIND_AUTO_CREATE);
- 作用:通过隐式意图,找到服务端对应的
Service
,发起绑定,触发服务端onBind
执行。
2. 服务端 onBind
返回 IBinder
服务端 IRemoteService
中,onBind
返回 AIDL 生成的 Stub
(本质是 IBinder
实现类 ):
public class IRemoteService extends Service {private IBinder iBinder = new IMyAidlInterface.Stub() { // Stub 实现了 IBinder@Overridepublic int add(int num1, int num2) throws RemoteException {return num1 + num2;}};@Overridepublic IBinder onBind(Intent intent) {return iBinder; // 返回 IBinder 实例}
}
- 说明:
Stub
是 AIDL 工具根据.aidl
文件自动生成的类,它继承Binder
并实现 AIDL 接口,用于跨进程通信时的 “桥梁”。
3. 客户端 onServiceConnected
回调
客户端通过 ServiceConnection
的 onServiceConnected
拿到服务端返回的 IBinder
,代码:
private ServiceConnection serviceConnection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {// 把 IBinder 转成 AIDL 接口代理aidlInterface = IMyAidlInterface.Stub.asInterface(service); }@Overridepublic void onServiceDisconnected(ComponentName name) {aidlInterface = null;}
};
重点:!!!!
一、asInterface(service) 的工作原理
1. 服务端返回的 service 参数
服务绑定成功后,服务端通过 onBind() 返回的 IBinder 对象(即 Stub 实例)
会被传递到客户端的 onServiceConnected() 回调中,作为 service 参数。
2. asInterface() 方法的源码逻辑
AIDL 自动生成的 Stub.asInterface() 方法大致如下:public static IMyAidlInterface asInterface(android.os.IBinder obj) {if (obj == null) {return null;}// 检查是否在同一进程(关键判断)android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);if (iin != null && iin instanceof IMyAidlInterface) {return (IMyAidlInterface) iin; // 同一进程:直接返回服务端的 Stub 本身}// 跨进程:创建并返回代理对象 Proxyreturn new IMyAidlInterface.Stub.Proxy(obj);
}同一进程(客户端和服务端在同一个 APK 中):
obj.queryLocalInterface() 返回服务端的 Stub 实例(因为 Stub 实现了 IInterface),
此时客户端直接拿到服务端的 Binder 对象,方法调用等效于本地调用。跨进程(客户端和服务端是不同 APK):
queryLocalInterface() 返回 null,此时创建 Proxy 对象,
它封装了底层的 Binder 通信逻辑
(将方法调用转为 transact() 请求,才开始进行数据复制那些内核态空间的操作),
让客户端通过代理间接访问服务端。二、为什么需要代理对象?
1. 跨进程通信的复杂性
客户端和服务端在不同进程,无法直接调用对方的方法。
代理对象(Proxy)的作用是:
将客户端的方法调用(如 add(5, 3))转换为 Binder 协议的 transact() 请求;
将参数打包为 Parcel 并发送给服务端;
接收服务端返回的结果并解包。2. 对开发者透明的封装
通过 asInterface() 返回代理对象,AIDL 隐藏了底层的跨进程通信细节,
让开发者感觉就像在调用本地对象的方法一样简单。
4. 客户端通过代理类调用接口!!!!!!!!!!!!!!
触发 Binder 驱动的实际数据传输和跨进程交互!!!!!!!
拿到 aidlInterface
(代理对象)后,直接调用 AIDL 定义的方法,比如:
try {int result = aidlInterface.add(5, 3); // 调用远程服务的 add 方法resultTv.setText("计算结果:" + result);
} catch (RemoteException e) {e.printStackTrace();
}
一、add(5, 3)
调用触发的完整流程
1. 客户端代理对象(Proxy
)处理调用
当执行 aidlInterface.add(5, 3)
时,实际调用的是 Stub.Proxy
类的 add()
方法(AIDL 自动生成):
// Proxy 类的 add 方法(伪代码)
@Override
public int add(int num1, int num2) throws RemoteException {// 1. 创建 Parcel 用于打包参数Parcel _data = Parcel.obtain();Parcel _reply = Parcel.obtain();int _result;try {// 2. 将参数写入 Parcel(用户空间)_data.writeInterfaceToken(DESCRIPTOR);_data.writeInt(num1);_data.writeInt(num2);3. 通过 Binder 发送请求到服务端(关键跨进程操作)!!!!!!!!!!!!!mRemote.transact(Stub.TRANSACTION_add, _data, _reply, 0);// 4. 从返回结果中读取数据(用户空间)_reply.readException();_result = _reply.readInt();} finally {_reply.recycle();_data.recycle();}return _result;
}
2. Binder 驱动的数据复制与映射
- 步骤 3 的
transact()
触发内核操作:- 客户端用户空间 → 内核空间:
Binder 驱动将_data
中的数据(5
和3
)从客户端进程的用户空间复制到内核缓存区(第一次内存拷贝)。 - 内核空间 → 服务端用户空间:
通过内存映射(mmap),服务端进程的用户空间直接与内核缓存区共享同一块物理内存,无需再次复制(零拷贝)。 - 服务端处理请求:
服务端的Stub
在onTransact()
中解析参数,调用真正的add()
方法计算结果(5 + 3 = 8
)。 - 结果返回:
结果(8
)从服务端用户空间写入内核缓存区(通过映射直接操作),再由 Binder 驱动复制到客户端的用户空间(第二次内存拷贝)。
- 客户端用户空间 → 内核空间:
3、Binder 高效性的核心
- 零拷贝优化:
通过内存映射(mmap),服务端可直接访问内核缓存区的数据,避免了传统 IPC(如 Socket)的多次拷贝(用户空间 → 内核空间 → 用户空间)。 - 两次拷贝(双向的,单项就只有一次拷贝):
Binder 实际只需要两次拷贝(客户端用户空间 → 内核空间 → 客户端用户空间),是 Android 中最高效的 IPC 机制之一。
而服务绑定(bindService()
)只是为这个过程建立通信通道,并不涉及实际业务数据的传输。
二、补充说明(帮你更深入理解)
- 同进程 vs 跨进程:如果客户端和服务端在同一个进程(比如调试时),
Stub.asInterface
会直接返回服务端的Stub
本身,不走跨进程通信逻辑(相当于本地接口调用);跨进程时才会返回Proxy
代理。 - 异常处理:跨进程调用可能因服务端崩溃、连接断开等抛
RemoteException
,所以客户端调用接口时要捕获异常,做容错处理(比如提示 “服务异常”)。 - AIDL 作用:帮我们自动生成
Stub
(服务端实现)和Proxy
(客户端代理)的模板代码,简化跨进程通信的 “编解码”“Binder 调用” 等复杂流程,让我们像写本地接口一样写跨进程逻辑 。
简单说,整个流程就是:客户端绑定服务 → 服务端返回 Binder → 客户端拿到代理 → 代理帮我们做跨进程调用 ,AIDL 主要是帮我们简化了 “代理创建”“数据编解码” 这些繁琐工作~