与简单的 C 函数调用相比,从 Java 调用 JNI 方法相当昂贵。HotSpot 通常执行以下大部分步骤来调用 JNI 方法:
- 创建堆栈帧。
- 根据 ABI 将参数移动到正确的寄存器或堆栈位置。
- 包装对 JNI 句柄的对象引用。
- 获取 和 对于静态方法,并将其作为附加参数传递。
JNIEnv*
jclass
- 检查是否应该调用跟踪函数。
method_entry
- 如果方法是 锁定对象监视器(如果该方法是 )。
synchronized
- 检查本机函数是否已链接。函数查找和链接执行得很懒惰。
- 将线程从切换到状态。
in_java
in_native
- 调用本机函数
- 检查是否需要安全点。
- 将线程返回到状态。
in_java
- 如果锁定,请解锁显示器。
- 通知。
method_exit
- 解包对象结果并重置 JNI 处理块。
- 处理 JNI 异常。
- 移除堆栈框架。
此过程的源代码可以在 SharedRuntime::generate_native_wrapper 中找到。
如您所见,开销可能很大。但在许多情况下,上述大多数步骤都不是必需的。例如,如果本机方法只是在字节数组上执行一些编码/解码,并且不引发任何异常,也不调用其他 JNI 函数。对于这些情况,HotSpot 有一个非标准(和未知)约定,称为 ,在这里讨论。Critical Natives