JNINativeMethod的参数解析

Android JNI 使用的数据结构JNINativeMethod详解
2009-04-02 15:53

Andoird 中使用了一种不同传统Java JNI的方式来定义其native的函数。其中很重要的区别是Andorid使用了一种Java 和 C 函数的映射表数组,并在其中描述了函数的参数和返回值。这个数组的类型是JNINativeMethod,定义如下:

 

typedef struct {
const char* name;
const char* signature;
void* fnPtr;
} JNINativeMethod;

Android JNI 使用的数据结构JNINativeMethod详解
2009-04-02 15:53

Andoird 中使用了一种不同传统Java JNI的方式来定义其native的函数。其中很重要的区别是Andorid使用了一种Java 和 C 函数的映射表数组,并在其中描述了函数的参数和返回值。这个数组的类型是JNINativeMethod,定义如下:

 

typedef struct {
const char* name;
const char* signature;
void* fnPtr;
} JNINativeMethod;

第一个变量name是Java中函数的名字。

第二个变量signature,用字符串是描述了函数的参数和返回值

第三个变量fnPtr是函数指针,指向C函数。

 

其中比较难以理解的是第二个参数,例如

"()V"

"(II)V"

"(Ljava/lang/String;Ljava/lang/String;)V"

 

实际上这些字符是与函数的参数类型一一对应的。

"()" 中的字符表示参数,后面的则代表返回值。例如"()V" 就表示void Func();

"(II)V" 表示 void Func(int, int);

 

具体的每一个字符的对应关系如下

 

字符 Java类型 C类型

V      void            void
Z       jboolean     boolean
I        jint              int
J       jlong            long
D      jdouble       double
F      jfloat            float
B      jbyte            byte
C      jchar           char
S      jshort          short

 

数组则以"["开始,用两个字符表示

 

[I       jintArray      int[]
[F     jfloatArray    float[]
[B     jbyteArray    byte[]
[C    jcharArray    char[]
[S    jshortArray   short[]
[D    jdoubleArray double[]
[J     jlongArray     long[]
[Z    jbooleanArray boolean[]

 

上面的都是基本类型。如果Java函数的参数是class,则以"L"开头,以";"结尾中间是用"/" 隔开的包及类名。而其对应的C函数名的参数则为jobject. 一个例外是String类,其对应的类为jstring

 

Ljava/lang/String; String jstring
Ljava/net/Socket; Socket jobject

 

如果JAVA函数位于一个嵌入类,则用$作为类名间的分隔符。

例如 "(Ljava/lang/String;Landroid/os/FileUtils$FileStatus;)Z"

第一个变量name是Java中函数的名字。

第二个变量signature,用字符串是描述了函数的参数和返回值

第三个变量fnPtr是函数指针,指向C函数。

 

其中比较难以理解的是第二个参数,例如

"()V"

"(II)V"

"(Ljava/lang/String;Ljava/lang/String;)V"

 

实际上这些字符是与函数的参数类型一一对应的。

"()" 中的字符表示参数,后面的则代表返回值。例如"()V" 就表示void Func();

"(II)V" 表示 void Func(int, int);

 

具体的每一个字符的对应关系如下

 

字符 Java类型 C类型

V      void            void
Z       jboolean     boolean
I        jint              int
J       jlong            long
D      jdouble       double
F      jfloat            float
B      jbyte            byte
C      jchar           char
S      jshort          short

 

数组则以"["开始,用两个字符表示

 

[I       jintArray      int[]
[F     jfloatArray    float[]
[B     jbyteArray    byte[]
[C    jcharArray    char[]
[S    jshortArray   short[]
[D    jdoubleArray double[]
[J     jlongArray     long[]
[Z    jbooleanArray boolean[]

 

上面的都是基本类型。如果Java函数的参数是class,则以"L"开头,以";"结尾中间是用"/" 隔开的包及类名。而其对应的C函数名的参数则为jobject. 一个例外是String类,其对应的类为jstring

 

Ljava/lang/String; String jstring
Ljava/net/Socket; Socket jobject

 

如果JAVA函数位于一个嵌入类,则用$作为类名间的分隔符。

例如 "(Ljava/lang/String;Landroid/os/FileUtils$FileStatus;)Z"

  • 3
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
JNINativeMethod 是一个结构体,它包含了 JNI 方法的名称、签名和函数指针。在 JNI 中,Java 方法可以被映射到本地 C/C++ 函数,JNINativeMethod 就是用来描述这样的映射关系的。 在 Java 中,声明一个 native 方法时需要使用 native 关键字,例如: ``` public native int myNativeMethod(String arg); ``` 在 C/C++ 中,实现一个本地方法需要先定义一个与之对应的 JNINativeMethod 结构体,例如: ``` JNIEXPORT jint JNICALL Java_MyClass_myNativeMethod(JNIEnv *env, jobject obj, jstring arg) { // 实现本地方法的代码 } ``` 其中,Java_MyClass_myNativeMethod 就是一个符合 JNI 命名规范的函数名,它由三部分组成:Java、类名和方法名,中间用下划线 _ 分隔,最后需要带上 JNIEnv 和 jobject 两个参数。 如果有多个本地方法需要实现,可以定义一个 JNINativeMethod 数组,将所有的方法都描述出来,例如: ``` JNINativeMethod methods[] = { {"myNativeMethod", "(Ljava/lang/String;)I", (void*)Java_MyClass_myNativeMethod}, {"anotherNativeMethod", "()V", (void*)Java_MyClass_anotherNativeMethod}, // ... }; ``` 在初始化 JNI 时,需要调用 RegisterNatives 函数将 JNINativeMethod 数组注册到对应的 Java 类中,例如: ``` jclass cls = env->FindClass("MyClass"); env->RegisterNatives(cls, methods, sizeof(methods) / sizeof(methods[0])); ``` 其中,FindClass 函数用于获取 Java 类的 jclass 对象,RegisterNatives 函数用于将 JNINativeMethod 数组注册到对应的 Java 类中。 这就是将 Java 的 native 方法转换为 JNINativeMethod 的基本流程。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值