zoukankan      html  css  js  c++  java
  • 技术转载:Jni学习四:如何编写jni方法

    转载:http://blog.chinaunix.net/u1/38994/showart_1099528.html


    一、概述:

    在这篇文章中将会简单介绍如何编制一些简单的JNI 方法。我们都知道JNI方法可以帮助我们调用用C/c++编写的函数,这样如果一项工作已经用C/c++语言实现的话,我们就可以不用花很大的力气再用JAVA语言对这一工作进行再实现,只要编制相应的JNI函数,就可以轻松实现JAVA语言对C/c++函数的调用,从而大大减轻程序开发人员的工作量。

    在这个项目中,我们编制了很多小实例,通过阅读,运行这些小实例,你可以轻松的学会如何编制JNI方法。这篇文档可以帮助你更好的理解及实现这些实例。

    现在让我们进入主题。首先,我们看一下这个项目的体系构架。该项目分为两部分,一部分用c语言是c语言的例子,另一部分是c++语言的例子。每部分都包含java,src源文件目录,以及一个Makefile文件。java目录中是需要调用JNI函数的JAVA源程序,含有后缀名.java。src 目录中含有JNI函数的实现代码,包括.c或.cpp文件和.h文件。Makefile文件是对 java 、src 目录下的文件进行编译组织进而生成可执行文件的文件。当Makefile文件执行以后还会生成以下子目录:lib , class ,bin目录 。lib 目录中包含项目中生成的静态函数库文件libJNIExamples.so,java程序所调用的JNI方法都是通过这个库来调用的。class 目录中包含由java目录下的.java 文件生成的.class文件。bin目录中是一个可执行的shell脚本文件。在执行该脚本的时候,项目所有程序实例的运行结果都将一并显示在屏幕上。


    具体执行步骤为:

    make

    cd bin

    ./run.sh


    下面来介绍一下在这个项目中所实现的实例:

       1. 如何调用标准C/c++中的函数--例如:printf(...)
       2. 如何调用C/c++中自定义的函数
       3. 如何在jni函数中访问java类中的对象实例域
       4. 如何在jni函数中访问java类中的静态实例域
       5. 如何在jni函数中调用java对象的方法
       6. 如何在jni函数中调用java类的静态方法
       7. 如何在jni函数中传递基本数据类型参数
       8. 如何在jni函数中传递对象类型参数
       9. 如何在jni函数中处理字符串
      10. 如何在jni函数中处理数组
      11. 处理jni函数中的返回值情况
      12. 在jni中实现创建java类对象


    二、基本步骤:

    在介绍这些例子之前,让我们先来看看编写jni方法所需要的基本步骤,这些实例都是用c来实例来讲解,至于c++的实例和c的实例区别不大,只要作稍微的修改即可,在文档的末尾我们将介绍这些内容:

    1、要想定义jni方法,首先得要在java语言中对这一方法进行声明(自然这一声明过程要在类中进行)

    声明格式如下:

    publicnativevoid print();   System.loadLibrary(“JNIExamples”);   }  

    jni 函数用关键字native方法声明。

    2、对该类的源文件进行编译使用javac命令,生成相应的.class文件。
    3、用javah -jni为函数生成一个在java调用和实际的c函数之间的转换存根,该存根通过从虚拟机栈中取出参数信息,并将其传递给已编译的C函数来实现转换。
    4、建立一个特殊的共享库,并从该共享库到处这个存根,在上面的例子中使用了System.loadLibrary,来加载libJNIExamples共享库。


    三、配置运行环境:

    在编写一个简单的jni函数之前我们必须配置相应的运行环境。jdk的配置在这里就不作介绍,这里主要说的是库的路径。当调用System.loadLibrary(..)时,编译器会到我们系统设置的库路径中寻找该库。修改路径的方法和修改任何环境变量的方法基本相同,只要在/etc/bash.bashrc目录下增加一行LD_LIBRARY_PATH=.:./lib:$(LD_LIBRARY_PATH)即可。也可以通过命令行export LD_LIBRARY_PATH=.:./lib:$(LD_LIBRARY_PATH)


    四、运行实例分析:

    1、实例一:在jni中调用标准c中自带的函数printf():

    下面以实例1为例来详细说明编写jni方法的详细过程。

    (1)、定义包含jni函数的类Print.java:
    {   /*********************************************************************** * the print() function will call the printf() funcion which is a ANSI c funciton * *************************************************************************/publicnativevoid print();       System.loadLibrary("JNIExamples");     }   }  


    在上面的实例中,使用public native void print();语句来定义了一个Print类的jni方法。并用Sysgem.loadLibrary(“JNIExamples”)语句来加载libJNIExamples.so库。注意:加载的语句一定要用static关键字声明在静态块中,以保证引用该类时该库始终被加载。

    (2)、对该类进行编译:javac Print.java。生成Print.class类,然后用javah 产生一个Print.h的头文件:javah Print。长生的Print.h文件格式如下:
    /* DO NOT EDIT THIS FILE - it is machine generated *//* Header for class Print */  JNIEXPORT void JNICALL Java_Print_print     (JNIEnv *, jobject);   }   

    其中的加粗字体为要实现的JNI函数生命部分。

    (3)、编写JNI函数的实现部分Print.c
    JNIEXPORT void JNICALL Java_Print_print (JNIEnv *env, jobject obj)   {     printf("example1:in this example a printf() function in ANSI C is called ");     printf("Hello,the output is generated by printf() function in ANSI C ");   }  

    在这个文件中实现了一个简单的Jni方法。该方法调用ANSI C 中的printf()函数,输出了两个句子。

    (4)、将本地函数编译到libJNIExamples.so的库中:
    使用语句:gcc -fPIC -I/usr/jdk1.5/include -I/usr/jdk1.5/include/linux -shared -o libJNIExamples.so Print.c。

    (5)、至此Jni函数已全部实现,可以在java代码中调用拉。
    在此我们使用一个简单的类来对实现的jni方法进行测试,下面是PrintTest.java的源代码部分:
    publicstaticvoid main(String[] args) {       Print p = new Print();       p.print();     }   }  

    (6)、对PrintTest.java进行编译执行得到如下结果:
    example1:in this example a printf() function in ANSI C is called
    Hello,the output is generated by printf() function in ANSI C .

    下面介绍的每个实例实现的步骤也都是按着上述步骤执行的。所以介绍时只介绍实现的关键部分。

    2、实例二、调用c 语言用户定义的函数(源程序为:java/Cfunction.java java/C_functionTest.java src/Cfunction.c src/Cfunction.h )
    当需要在java程序中调用用c所实现的函数是,需要在需要调用该c函数的类中定义一个jni方法,在该jni方法中去调用该c函数,相当于用java方法把c函数封装起来,以供java程序调用。
    在实例二中我们简单定义了一个printHello()函数,该函数的功能只是输出一句话,如果要在java程序中调用该函数,只需在jni函数中调用即可,和调用ANSI C中自带的prinf()函数没有任何区别。

    3、实例三、在jni函数中访问java类中的对象实例域(源程序为:java/CommonField.java java/CommonFieldTest.java src/CommonField.c src/CommonField.h )
    jni函数的实现部分是在c 语言中实现的,如果它想访问java中定义的类对象的实例域需要作三步工作,
    (1)调用GetObjectClass()函数得到该对像的类,该函数返回一个jclass类型值。
    (2)调用GetFieldID()函数得到要访问的实例域在该类中的id。
    (3)调用GetXXXField()来得到要访问的实例域的值。其中XXX和要访问的实例域的类型相对应。
    在jni中java 编程语言和c 语言数据类型的对应关系为java原始数据类型前加 'j' 表示对应c语言的数据类型例如boolean 为jboolean ,int 为 jint,double 为jdouble等。对象类型的对应类型为jobject。
    在本实例中,您可以看到我们在java/CommonField.java 中定义了类CommonField类,其中包含int a , int b 两个实例域,我们要在jni函数getCommonField()中对这两个域进行访问和修改。你可以在 src/CommonField.c中找到该函数的实现部分。
    以下语句是对该域的访问(以下代码摘自:src/CommonField.c):

    jclass class_Field = (*env)->GetObjectClass(env,obj);   jfieldID fdA = (*env)->GetFieldID(env,class_Field,"a","I");   jfieldID fdB = (*env)->GetFieldID(env,class_Field,"b","I");   jint valueA = (*env)->GetIntField(env,obj,fdA);   jint valueB = (*env)->GetIntField(env,obj,fdB);  


    在jni中对所有jni函数的调用都要用到env指针,该指针也是每一个本地方法的第一个参数,他是函数指针表的指针,所以,必须在每一个jni调用前面加上(*env)->GetObjectClass(env,obj)函数调用返回obj对像的类型,其中obj 参数表示要你想要得到类型的类对象。
    jfieldID GetFieldID(JNIEnv *env,jclass cl, const char name[], const char sig[]) 该函数返回一个域的标识符name 表示域名,sig表示编码的域签名。所谓编码的签名即编码类型的签名在上例中类中的a实例域为int 型,用"I”来表示,同理"B” 表示byte ,"C” 表示 char , “D”表示 double ,”F” 表示float,“J”表示long, “S” 表示short , “V” 表示void ,”Z”表示 boolean类型。
    GetIntField(env,obj,fdA),用来访问obj对象的fdA域,如果要访问的域为double类型,则要使用GetDoubleField(env,obj,fdA)来访问,即类型对应GetXXXField中的XXX。

    以下函数用来修改域的值:

    (*env)->SetIntField(env,obj,fdA,109);   (*env)->SetIntField(env,obj,fdB,145);  

    这和获得域的值类似,只是该函数多了一个要设置给该域的值参数。
    访问对象实例域的相关函数如下:
    jfieldID GetFieldID(JNIEnv *env, jclass cl, const char name[], const char sig[])
    该函数返回一个域的标识符。各参数含义如下:
    env JNI 接口指针;cl 类对象 ; name 域名; sig 编码的域签名

    XXX GetXXXField(JNIEnv *env, jobject obj, jfieldID id)
    该函数返回域的值。域类型XXX是Object, Boolean, byte, char , short, int ,long ,float, double 中类型之一。
    参数 env JNI借口指针;obj为域所在对象;id为域的标识符。
    void SetXXXField(JNIEnv *env,jobject obj, jfieldID id, XXX value)

    该函数用于设置域的值。XXX的含义同上,
    参数中env, obj , id 的含义也同上,value 值为将要设置的值。

    4、实例四:在jni函数中访问类的静态实例域 (java/Field.java java/FieldTest.java src/Field.c src/Field.h)

    因为静态实例域并不属于某个对象,而是属于一个类,所以在要访问静态实例域时,和访问对象的实例域不同,它所调用的函数是(以实例四来说明,一下代码摘自src/Field.c):
    jclass class_Field = (*env)->FindClass(env,"Field");   jfieldID fdA = (*env)->GetStaticFieldID(env,class_Field,"a","I");   jint valueA = (*env)->GetStaticIntField(env,class_Field,fdA);   (*env)->SetStaticIntField(env,class_Field,fdA,111);  

    由于没有对象,必须使用FindClass代替GetObjectClass来获得类引用。在FindClass()的第二个参数是类的编码签名,类的编码签名和基本类型的编码签名有所不同,如果类在当前包中,就直接是类的名称,如果类不在当前包中则要加入该类的详细路径:例如String类在java.lang包中,则String的签名要写成( Ljava/lang/String;),其中的(L和;)是不可少的,其中(;)是表达是的终止符。其他三个函数和访问对象数据域基本没什么区别。

    5、实例五:在jni函数中调用java对象的方法(java/CommonMethod.java java/CommonMethodTest.java src/CommonMehod.c src/CommonMethod.h )

    在jni函数中我们不仅要对java对象的数据域进行访问,而且有时也需要调用java中类对象已经实现的方法,实例五就是关于这方面的实现的。在src/CommonMethod.c中我们可以找到下面的代码:

    JNIEXPORT void JNICALL Java_CommonMethod_callMethod   (JNIEnv *env, jobject obj, jint a, jstring s)   {     printf("example 5:in this example,a object's method will be called ");     jclass class_CommonMethod = (*env)->GetObjectClass(env,obj);     jmethodID md = (*env)->GetMethodID(env,class_CommonMethod,"print","(ILjava/lang/String;)V");     (*env)->CallVoidMethod(env,obj,md,a,s);   }  


    该代码部分展示了如何实现对java类对象函数的调用过程。从以上代码部分我们可以看到,要实现该调用需要有三个步骤,调用三个函数
    jclass class_CommonMethod = (*env)->GetObjectClass(env,obj);   jmethodID md = (*env)->GetMethodID(env,class_CommonMethod,"print","(ILjava/lang/String;)V");   (*env)->CallVoidMethod(env,obj,md,a,s);  

    GetObjectClass(...)函数获得要调用对象的类;GetMethodID(...)获得要调用的方法相对于该类的ID号;CallXXXMethod(...)调用该方法。
    在编写该调用过程的时候,需要注意的仍然是GetMethodID(...)函数中编码签名的问题,在该实例中,我们要做的是找到CommonMethod类的print(int a, String s)方法,该方法打印整数a,和字符串s 的直。在函数的编码签名部分(该部分以加粗、并加有下划线)GetMethodID(env,class_CommonMethod,"print","(ILjava/lang/String;)V"); 从左往右可以查看,括号中的内容为要调用方法的参数部分内容,I表示第一个参数为int类型,“Ljava/lang/String;”表示第二个参数为String类型,V表示返回值类型为空void,如果返回值类型不为空,则使用相应的类型签名。返回值类型是和下面将要使用的调用该方法的函数CallXXXMethod(...)相关联的,该函数的xxx要用相应的类型来替换,在此实例中为void,如果返回值类型为int类型则调用该方法的函数就为CallIntMethod(...)。


    6、实例六:在jni函数中调用java类的静态方法(java/Method.java java/MethodTest.java src/Method.h src/Method.c)

    实例五中介绍了如何调用类对象的方法,在此实例中我们将介绍如何调用java类的静态方法在此实例中我们在/java/Method.java中定义了静态方法:

    public static void print() {
      System.out.println("this is a static method of class Method");
    }

    该函数的功能就是打印字符串“ this is a static method of class Method”;
    我们在src/Method.c中实现了对该方法调用的jni函数:
    JNIEXPORT void JNICALL Java_Method_callMethod   (JNIEnv *env, jobject obj)   {     printf("example 6:in this example, the class's static method will be called ");     jclass class_Method = (*env)->FindClass(env,"Method");     jmethodID md = (*env)->GetStaticMethodID(env,class_Method,"print","()V");     (*env)->CallStaticVoidMethod(env,class_Method,md);   }  


    和实例五不同的是,我们要调用的三个函数变为:
    FindClass(...)、GetStaticMethodID(...)、CallStaticVoidMethod(...)。
    其中的机制和实例五是一样的。再次就不做过多的介绍。

    7、实例七:jni函数中传递基本数据类型参数(java/Basic.java java/BasicTest.java src/Basic.c src/Basic.h) 在java/Basic.java中,我们定义了一个public native void raiseValue(int a)函数,该函数将打印使value的值增加a,并打印原来的value和新的value值。
    在src/Basic.c中给出了该jni函数的实现部分。
    JNIEXPORT void JNICALL Java_Basic_raiseValue   (JNIEnv *env, jobject obj, jint a)   {     printf("example 7: in this example, a integer type parament will be passed to the jni method ");     jclass class_Basic = (*env)->GetObjectClass(env,obj);     jfieldID fd = (*env)->GetFieldID(env,class_Basic,"value","I");     jint v = (*env)->GetIntField(env,obj,fd);     v = v+a;     (*env)->SetIntField(env,obj,fd,v);   }  

    在此函数实现中,因为要访问Basic类中的value域,所以调用了GetObjectClass(...), GetFieldID(...), GetIntField(...)函数获取value值,下面一步的 “ = v+a; ”说明,传递基本类型参数的处理方式和在c语言中的基本数据类型的处理无异。


    8、实例八:在jni函数中传递对象类型参数(java/Book.java java/BookTest.java src/BookTest.c src/BookTest.h)

      在该实例中演示了在jni函数中传递对象函数的过程。

      我们在该实例中定义了一个类Book
        total_page = t;     }   publicint getTotalPage() {     }   publicint getCurrentPage() {     }       current_page++;     }   }  

    然后我们在java/BookTest.java中定义jni函数
    public native void bookCurrentStatus(Book b);
    该函数需要一个Book类型的参数,并返回该参数的当前状态,包括该书一共有多少页的total_page,以及当前页current_page。函数的实现部分为(src/BookTest.c)
    JNIEXPORT void JNICALL Java_BookTest_bookCurrentStatus   (JNIEnv *env, jobject this_obj, jobject obj)   {     printf("example 8: in this example, a object parament will be passed to the jni method。 ");     jclass class_book = (*env)->GetObjectClass(env,obj);     jmethodID id_getTotal = (*env)->GetMethodID(env,class_book,"getTotalPage","()I");     jmethodID id_getCurrent = (*env)->GetMethodID(env,class_book,"getCurrentPage","()I");     jint total_page = (*env)->CallIntMethod(env,obj,id_getTotal);     jint current_page = (*env)->CallIntMethod(env,obj,id_getCurrent);     printf("the total page is:%d and the current page is :%d ",total_page,current_page);   }  

    该函数包含三个参数(JNIEnv *env, jobject this_obj, jobject obj) ,第二个jobject this_obj参数表示当前的jni 函数所属于的类对象,第三个jobject obj参数表示传递的参数Book类型的类对象。
    对于实现部分,基本和实例五--调用java类对象的方法中的操作相同,就不作详解。

    9、实例九:在jni函数中处理字符串(java/Str.java java/StrTest.java src/Str.c src/Str.h)
    在该实例中我们讲解如何传递、处理字符串参数。
    在java/Str.java中我们定义了一个 printString(String s) 的方法,用来处理字符串参数。
    在src/Str.c中我们可以看到该函数的实现部分:
    JNIEXPORT void JNICALL Java_Str_printString   (JNIEnv *env, jobject obj, jstring s)   {     printf("example 9: in this example, a String object parament will be passed to the jni method. ");     string = (char*)(*env)->GetStringUTFChars(env,s,NULL);     printf("%s is put out in native method ",string);     (*env)->ReleaseStringUTFChars(env,s,(jbyte*)string);   }  

    实现过程中调用了两个函数:GetStringUTFChars(...)、 ReleaseStringUTFChars(...)。
    GetStringUTFChars(...)用来获取String对象的字符串,并将其抓那还为char*类型,这应该字符串就可以在c语言中进行处理拉。ReleaseStringUTFChars(...)用于当该字符串使用完成后,将其进行垃圾回收。记住,当使用完字符串时一定不要忘记调用该函数。

    10、实例十:在jni函数中处理数组(java/Arr.java java/ArrTest.java src/Arr.c src/Arr.h)
    java中所有的数组类型都有相对应的c语言类型,其中jarray类型表示一个泛型数组
    boolean[] --jbooleanArray byte[]--jbyteArray char[]--jcharArary
    int[]---jcharArray short[]---jshortArray long[]---jlongArray float[]--jfloatArray
    double[]—-jdoubleArray Object[]--- jobjectArray。当访问数组时,可以通过GetObjectAraryElement和SetObjectArrayElement方法访问对象数组的元素。
    而对于一般类型数组,你可以调用GetXXXAraryElements来获取一个只想数组起始元素的指针,而当你不在使用该数组时,要记得调用ReleaseXXXArrayElements,这样你所作的改变才能保证在原始数组里得到反映。当然如果你需要得到数组的长度,可以调用GetArrayLength函数。
    在本实例中,我们在Arr.java中定义一个本地方法:print(int intArry[]),该函数的功能为对该数组进行输出,在src/Arr.c中我们可以看到该方法的实现过程如下:
    JNIEXPORT void JNICALL Java_Arr_print   (JNIEnv *env, jobject obj, jintArray intArray)   {     printf("example 10:in this example, a array parament will be passed to the jni method. ");     arr = (*env)->GetIntArrayElements(env,intArray,NULL);   //n = (*env)->GetArrayLength(env,intArray);   printf("the native method output the int array ");   for( i = 0;i<(*env)->GetArrayLength(env,intArray);i++)     {       printf("%d ",arr[i]);     }     (*env)->ReleaseIntArrayElements(env,intArray,arr,0);   }  

    我们在此调用了GetIntArrayElements(...)来获取一个指向intArray[]数组第一个元素的指针。
    用getArrayLength(..)函数来得到数组的长度,以方便数组遍历时使用。最后应用ReleaseArrayElements(...)函数来释放该数组指针。

    11、实例十一:在jni中的返回值问题(java/ReturnValue.java java/ReturnValueTest.java java/BookClass.java src/ReturnValue.c src/ReturnValue.h)
    在java/ReturnValue类中定义了三个jni方法: returnInt(),returnString() ,returnObject()
    三个方法,分别返回int , String , Object 类型的值。
    其在src/ReturnValue.c中的实现分别为:

    JNIEXPORT jint JNICALL Java_ReturnValue_returnInt   (JNIEnv *env, jobject obj)   {     jclass class_ReturnValue = (*env)->GetObjectClass(env,obj);     jfieldID fd = (*env)->GetFieldID(env,class_ReturnValue,"value","I");     jint v = (*env)->GetIntField(env,obj,fd);   }      * Signature: ()Ljava/lang/String; JNIEXPORT jstring JNICALL Java_ReturnValue_returnString   (JNIEnv *env, jobject obj)   {     printf("example 11: in this example, the int and object of return value will be proceeding ");     jclass class_ReturnValue = (*env)->GetObjectClass(env,obj);     jfieldID fd = (*env)->GetFieldID(env,class_ReturnValue,"name","Ljava/lang/String;");     jstring jstr = (jstring)(*env)->GetObjectField(env,obj,fd);   }      * * Method: returnObject JNIEXPORT jobject JNICALL Java_ReturnValue_returnObject   (JNIEnv *env, jobject obj)   {     jclass class_ReturnValue = (*env)->GetObjectClass(env,obj);     jfieldID fd = (*env)->GetFieldID(env,class_ReturnValue,"myBook","LBookClass;");     jobject jbook = (jstring)(*env)->GetObjectField(env,obj,fd);   }  

    在这里分别涉及到了对java类对象的一般参数,String参数,以及Object参数的访问。


    12、实例十二:在jni中创建java类对象:(java/Test.java src/CreateObj.c src/CreateObj.h)

    如果想要在jni函数创建java类对象则要引用java 类的构造器方法,通过调用NewObject函数来实现。
    NewObject函数的调用方式为:
    jobject obj_new = (*env)->NewObject(env,class, methodid, paraments);
    在该实例中,我们在java/Test.java 中定义了Book1类,要在CreateObj类的modifyProperty() jni方法中创建该类对象。我们可以在src/CreateObj.c中看到该jni方法创建对象的过程:
    jobject book;   jclass class_book;   jmethodID md_book;   class_book = (*env)->FindClass(env,"LBook1;");   md_book = (*env)->GetMethodID(env,class_book,"<init>","(IILjava/lang/String;)V");   book = (*env)->NewObject(env,class_book,md_book,100,1,"huanghe");  


    在创建对象的过程中可以看到,要创建一个java类对象,首先需要得到得到使用FindClass函数得到该类,然后使用GetMethodID方法得到该类的构造器方法id,主义在此时构造器的函数名始终为:"”,其后函数的签名要符合函数签名规则。在此我们的构造器有三个参数:int , int, String.
    并且其返回值类型要永久为空,所以函数签名为:"(IILjava/lang/String;)V"
    然后我们调用NewObject()函数来创建该类的对象,在此之后就可以使用该对象拉。

    以上内容介绍的是jni函数c语言的实现实例。如果想要使用c++的实例,我们只需要把其中的每一个函数调用过程作稍微的修改:
    例如:(*env)->NewObject(env,class_book,md_book,100,1,”huanghe”);
    修改为:(env)->NewObject(class_book,md_book,100,1,”huanghe”);
    即修改(*env)为(env)再把参数中的env去掉。然后把所有c的函数改为c++的函数就OK拉。
    具体情况可以去查看我们的c++实例代码.
  • 相关阅读:
    Backbone学习记录(6)
    Backbone学习记录(5)
    Backbone学习记录(4)
    PHP中抽象类,接口定义
    php和js中json的编码和解码
    jquery中 dom对象与jQuery对象相互转换
    js post跳转
    javascript中的三种弹窗
    出现多个sessid
    php中cookie的操作
  • 原文地址:https://www.cnblogs.com/sharecenter/p/5621110.html
Copyright © 2011-2022 走看看