提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
提示:这里可以添加本文要记录的大概内容:
例如:相信很多java程序员对native方法和普通方法是如何运行以及底层原理很好奇.这篇文章是对此机制做一个技术分享。
提示:以下是本篇文章正文内容,下面案例可供参考
示例:查看java编译以后的方法层面的字节码,如下是普通方法的界面查找实例。
代码如下(示例):
public class Demo{
public static void main(String[] args) {
int sum = sum(1,2);
System.out.println(sum);
}
public static int sum(int a,int b){
return a+b;
}
}
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=2, args_size=1
0: iconst_1
1: iconst_2
2: invokestatic #2 // Method sum:(II)I
5: istore_1
6: getstatic #3 // Field java/lang/System.out:Ljava/io/PrintStream;
9: iload_1
10: invokevirtual #4 // Method java/io/PrintStream.println:(I)V
13: return
public class Demo{
static{
System.loadLibrary("CH347DLLA64.dll");
}
public static void main(String[] args) {
Demo demo = new Demo();
int sum = demo.sum(1,2);
System.out.println(sum);
}
public native int sum(int a,int b);
}
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=3, locals=3, args_size=1
0: new #2 // class Demo
3: dup
4: invokespecial #3 // Method "":()V
7: astore_1
8: aload_1
9: iconst_1
10: iconst_2
11: invokevirtual #4 // Method sum:(II)I
14: istore_2
15: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;
18: iload_2
19: invokevirtual #6 // Method java/io/PrintStream.println:(I)V
22: return
public native int sum(int, int);
descriptor: (II)I
flags: ACC_PUBLIC, ACC_NATIVE
static {};
descriptor: ()V
flags: ACC_STATIC
Code:
stack=1, locals=0, args_size=0
0: ldc #7 // String CH347DLLA64.dll
2: invokestatic #8 // Method java/lang/System.loadLibrary:(Ljava/lang/String;)V
5: return
LineNumberTable:
line 3: 0
line 4: 5
}
代码如下(示例):
CASE(_invokevirtual):
CASE(_invokespecial):
CASE(_invokestatic): {
u2 index = Bytes::get_native_u2(pc+1);
ConstantPoolCacheEntry* cache = cp->entry_at(index);
...
istate->set_callee(callee);
//方法中调用设置方法的入口点
istate->set_callee_entry_point(callee->from_interpreted_entry());
#ifdef VM_JVMTI
if (JvmtiExport::can_post_interpreter_events() && THREAD->is_interp_only_mode()) {
istate->set_callee_entry_point(callee->interpreter_entry());
}
#endif /* VM_JVMTI */
istate->set_bcp_advance(3);
UPDATE_PC_AND_RETURN(0); // I'll be back...
}
}
void set_interpreter_entry(address entry) { _i2i_entry = entry; _from_interpreted_entry = entry; }
void Method::link_method(methodHandle h_method, TRAPS) {
// _i2i_entry 先不关注,这个解析器和即时编译转换的入口点
if (_i2i_entry != NULL) return;
assert(_adapter == NULL, "init'd to NULL" );
assert( _code == NULL, "nothing compiled yet" );
// Setup interpreter entrypoint
assert(this == h_method(), "wrong h_method()" );
//重点就在这里,这次设置方法入口点
address entry = Interpreter::entry_for_method(h_method);
assert(entry != NULL, "interpreter entry must be non-null");
set_interpreter_entry(entry);
}
}
enum MethodKind {
... //例程类型
native, //native method
native_synchronized, //
...
};
address AbstractInterpreterGenerator::generate_method_entry(
AbstractInterpreter::MethodKind kind) {
...
case Interpreter::native:
entry_point = ((InterpreterGenerator*) this)->generate_native_entry(false); //native方法入例程
break;
if (entry_point == NULL)
entry_point = ((InterpreterGenerator*) this)->generate_normal_entry(false);//普通方法入例程
return entry_point;
}
int CppInterpreter::native_entry(Method* method, intptr_t UNUSED, TRAPS) {
**// 获取native方法对应的c的函数地址**
address function= method->native_function();
//此处是修改当前当前的执行状态
ThreadStateTransition::transition_from_java(thread, _thread_in_native);
//实际是通过ffi_call(Foreign Function Interface)库函数调用实际的c方法,按照如参数顺序和参数类型封装并调用将返回值进行封装.
intptr_t result[4 - LogBytesPerWord];
ffi_call(handler->cif(), (void (*)()) function, result, arguments);
...
// 其中涉及安全域,切换线程状态会做对应的安全性检查.
if (SafepointSynchronize::do_call_back() ||
thread->has_special_condition_for_native_trans()) {
JavaThread::check_special_condition_for_native_trans(thread);
CHECK_UNHANDLED_OOPS_ONLY(thread->clear_unhandled_oops());
}
最终native方法会根据规范在动态链接库.dll(win)或.so(linux)查找方法的地址 function ,然后使用ffi_call库将参数封装,调用动态库的函数的.以上是native调用的整个过程.