在前面两个函数的介绍里,已经找到要执行方法的类,要执行的方法ID,那么接着下来要做的事情,就是执行方法,其实就是解释Java程序的字节码。那么Davlik虚拟机是怎么样执行方法的代码呢?又是怎么样解释这些字节码指令呢?因此,就先来分析函数CallStaticVoidMethod的代码,直接拿这个函数名称在源程序里查找,是查找不到的,因为代码里并没有直接使用这个名称来声明函数,其实是使用宏来实现的,如下:


CALL_STATIC(void,Void, , , false);


从上面可以看到,这是一个宏定义CALL_STATIC生成的函数,直接查找是不行的。接着来看一看这个宏到底是怎么样定义的,如下:

/*

* Call a static method.

*/

#defineCALL_STATIC(_ctype, _jname, _retfail, _retok, _isref) \

static_ctype CallStatic##_jname##Method(JNIEnv*env, jclass jclazz, \

jmethodID methodID, ...) \

{

这个函数输入的参数有4个,第一个参数envJNI的运行环境;第二个参数jclazz是方法的类对象;第三个参数methodID是运行的方法ID;第四个参数是可变输入参数。

\

UNUSED_PARAMETER(jclazz); \

JNI_ENTER(); \

JValue result; \

va_list args; \

va_start(args, methodID); \

dvmCallMethodV(_self,(Method*)methodID, NULL, true, &result, args);\

这行代码是调用dvmCallMethodV函数来执行方法的代码。


va_end(args); \

if(_isref && !dvmCheckException(_self)) \

result.l =addLocalReference(env, result.l); \

JNI_EXIT(); \

return_retok; \

} \



下面这个方法是带可变参数的调用。

static_ctype CallStatic##_jname##MethodV(JNIEnv*env, jclass jclazz, \

jmethodID methodID,va_list args) \

{ \

UNUSED_PARAMETER(jclazz); \

JNI_ENTER(); \

JValue result; \

dvmCallMethodV(_self,(Method*)methodID, NULL, true, &result, args);\

if(_isref && !dvmCheckException(_self)) \

result.l =addLocalReference(env, result.l); \

JNI_EXIT(); \

return_retok; \

} \


下面这个方法是有数组参数的调用。

static_ctype CallStatic##_jname##MethodA(JNIEnv*env, jclass jclazz, \

jmethodID methodID,jvalue* args) \

{ \

UNUSED_PARAMETER(jclazz); \

JNI_ENTER(); \

JValue result; \

dvmCallMethodA(_self,(Method*)methodID, NULL, true, &result, args);\

if(_isref && !dvmCheckException(_self)) \

result.l =addLocalReference(env, result.l); \

JNI_EXIT(); \

return_retok; \

}

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐