Chendd's Blog

世界上没有什么事情是一行代码不能解决的。如果有,那就两行。

深入了解android平台的jni(一)

       android中很多Java类都具有native接口,这些接口由本地实现,然后注册到系统中。
主要的JNI代码放在以下的路径中:frameworks/base/core/jni/,这个路径中的内容被编译成库 libandroid_runtime.so,被放置在目标系统的/system/lib目录下。此外,android还有其他的 JNI库。JNI中的各个文件,实际上就是普通的C++源文件. 如果要深入了解android framework层,则必须Android Native层运行及开发机制. 这里先介绍一些native的基础知识

1、接口定义

       _JNIEnv定义了一个虚拟机的接口,通过这个接口可以访问虚拟机的所有功能: 1)分配对象(AllocObject/NewObject),并且控制对象的引用计数(NewGlobalRef/DeleteGlobalRef/DeleteLocalRef/IsSameObject/NewLocalRef)。 2)获取类的定义(FindClass),并通过类的定义来获取获取类得方法和成员的ID(GetMethodID/GetFieldID) 3)通过方法ID调用类的普通方法(CallObjectMethod)和静态方法(CallStaticObjectMethod) 4)通过成员ID获取和设置类的普通成员(GetObjectField/SetObjectField)和静态成员(GetStaticObjectField/SetStaticObjectField)

下面是比较常用的方法:

1.1查找该类:

jclass xxx = (*env)->FindClass(env, “Lclass_name;”);

1.2取得方法的id:

jmethodID xxx = (*env)->GetMethodID(env, jclass, methodName, “(M)N”);

1.3查找需要调用的该类的方法:

jmethodID xxx = (*env)->GetMethodID(env, jclass, “(M)N” );

1.4取得静态方法的id

jmethodID xxx = (*env)->GetStaticMethodID(env,jclass, methodName,“(M)N”)

1.5初始化该类的实例:

jobject xxx = (*env)->NewObject(env, jclass, jmethodID );

1.6调用实例的某方法:

(*env)->CallObjectMethod(env, jobject, jmethodID, [parameter1, parameter2,…] );

1.7释放实例:

(*env)->DeleteLocalRef(env, xxx);

1.8取得成员变量的id

jfieldID xxx = (*env)->GetFieldID(env,jclass ,jfieldID,jfieldType)

1.9取得静态成员变量的id

jfieldID xxx = GetStaticFieldID(env,jclass ,jfieldID,jfieldType) JNIENV - java的运行环境 jobject - 代表java的instance jclass - 代表java的类

2、函数与属性签名

       在GetMethodID和GetFieldID这两个函数中,最后一个参数都是签名字符串,用来标示java函数和成员的唯一性。 因为java中存在重载函数,所以一个函数名不足以唯一指定一个函数,这时候就需要签名字符串来指定函数的参数列表和返回值类型了。 函数签名是一个字符串:"(M)N" 括号中的内容是函数的参数类型,括号后面表示函数的返回值。

3、JNI 类型签名

       “(M)N”,这里的M和N指的是该函数的输入和输出参数的类型签名(Type Signature)。 具体的每一个字符的对应关系如下:

字符–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” 举例说明"(M)N"的含义,例如:

(I)V 带一个int 类型的参数,返回值类型为void

()D 没有参数,返回double

Comments