http://blog.csdn.net/qinjuning/article/details/7607214

2024-05-16 08:48

本文主要是介绍http://blog.csdn.net/qinjuning/article/details/7607214,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文原创,转载请注明出处:http://blog.csdn.net/qinjuning




         在掌握了JNI函数的使用和相关类型的映射后,以及知晓何利用javah工具生成对应的jni函数以及如何生成动态

    链接库 (windos下就是.dll库,Linux就是.so库了,不懂在Window下生成dll动态库的,具体流程可看我的这篇博客:

   《Android中JNI的使用之一:Java原生JNI的使用、javah指令的使用以及图解教材》)。即可掌握JNI的使用了了。


        总的来说,JNI是不难的。通过前面的学习相信你应该有所了解。今天,我们从几个简单的小例子,来对JNI进行下实战训练。

     可都是些小例子,耐心看咯。

 

        主要操作内容,包括如下几个部分:


               1、在Native层返回一个字符串

               2、从Native层返回一个int型二维数组(int a[ ][ ]) 

               3、从Native层操作Java层的类: 读取/设置类属性

               4、在Native层操作Java层的类:读取/设置类属性、回调Java方法 

               5、从Native层返回一个复杂对象(即一个类咯)

               6、在Java层传递复杂对象至Native层

               7、从Native层返回Arraylist集合对象


      广而告知,这些操作就是简单的利用一些JNI函数即实现了。so easy 。


 一、在Native层返回一个字符串

       Java层原型方法:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     public native void getAJNIString();  
  4.     ...  
  5. }     

       Native层该方法实现为 :

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    getAJNIString 
  4.  * Signature: ()Ljava/lang/String; 
  5.  */   
  6. //返回字符串  
  7. JNIEXPORT jstring JNICALL Java_com_feixun_jni_HelloJni_getAJNIString(JNIEnv * env, jobject obj)  
  8. {  
  9.     jstring str = env->newStringUTF("HelloJNI");  //直接使用该JNI构造一个jstring对象返回  
  10.     return str ;  
  11. }  

 

二、在Native层返回一个int型二维数组(inta[ ][ ])

    Java层原型方法:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     //参数代表几行几列数组 ,形式如:int a[dimon][dimon]  
  4.     private native int[][] getTwoArray(int dimon) ;   
  5.     ...  
  6. }     

       Native层该方法实现为 :

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    getTwoArray 
  4.  * Signature: (I)[[I 
  5.  */  
  6. //通过构造一个数组的数组, 返回 一个二维数组的形式  
  7. JNIEXPORT jobjectArray JNICALL Java_com_feixun_jni_HelloJni_getTwoArray  
  8.   (JNIEnv * env, jobject object, jint dimion)  
  9. {  
  10.       
  11.     jclass intArrayClass = env->FindClass("[I"); //获得一维数组 的类引用,即jintArray类型  
  12.     //构造一个指向jintArray类一维数组的对象数组,该对象数组初始大小为dimion  
  13.     jobjectArray obejctIntArray  =  env->NewObjectArray(dimion ,intArrayClass , NULL);  
  14.   
  15.     //构建dimion个一维数组,并且将其引用赋值给obejctIntArray对象数组  
  16.     forint i = 0 ; i< dimion  ; i++ )  
  17.     {  
  18.         //构建jint型一维数组  
  19.         jintArray intArray = env->NewIntArray(dimion);  
  20.   
  21.         jint temp[10]  ;  //初始化一个容器,假设 dimion  < 10 ;  
  22.         forint j = 0 ; j < dimion ; j++)  
  23.         {  
  24.             temp[j] = i + j  ; //赋值  
  25.         }  
  26.           
  27.         //设置jit型一维数组的值  
  28.         env->SetIntArrayRegion(intArray, 0 , dimion ,temp);  
  29.         //给object对象数组赋值,即保持对jint一维数组的引用  
  30.         env->SetObjectArrayElement(obejctIntArray , i ,intArray);  
  31.   
  32.         env->DeleteLocalRef(intArray);  //删除局部引用  
  33.     }  
  34.   
  35.     return   obejctIntArray; //返回该对象数组  
  36. }  


 三、在Native层操作Java层的类 :读取/设置类属性


     Java层原型方法:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     //在Native层读取/设置属性值  
  4.     public native void native_set_name() ;  
  5.     ...  
  6.       
  7.     private String name = "I am at Java" ; //类属性  
  8. }     

    Native层该方法实现为 :

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    native_set_name 
  4.  * Signature: ()V  
  5.  */  
  6. //在Native层操作Java对象,读取/设置属性等  
  7. JNIEXPORT void JNICALL Java_com_feixun_jni_HelloJni_native_1set_1name  
  8.   (JNIEnv *env , jobject  obj )  //obj代表执行此JNI操作的类实例引用  
  9. {  
  10.    //获得jfieldID 以及 该字段的初始值  
  11.    jfieldID  nameFieldId ;  
  12.   
  13.    jclass cls = env->GetObjectClass(obj);  //获得Java层该对象实例的类引用,即HelloJNI类引用  
  14.   
  15.    nameFieldId = env->GetFieldID(cls , "name" , "Ljava/lang/String;"); //获得属性句柄  
  16.   
  17.    if(nameFieldId == NULL)  
  18.    {  
  19.        cout << " 没有得到name 的句柄Id \n;" ;  
  20.    }  
  21.    jstring javaNameStr = (jstring)env->GetObjectField(obj ,nameFieldId);  // 获得该属性的值  
  22.    const char * c_javaName = env->GetStringUTFChars(javaNameStr , NULL);  //转换为 char *类型  
  23.    string str_name = c_javaName ;    
  24.    cout << "the name from java is " << str_name << endl ; //输出显示  
  25.    env->ReleaseStringUTFChars(javaNameStr , c_javaName);  //释放局部引用  
  26.   
  27.    //构造一个jString对象  
  28.    char * c_ptr_name = "I come from Native" ;  
  29.      
  30.    jstring cName = env->NewStringUTF(c_ptr_name); //构造一个jstring对象  
  31.   
  32.    env->SetObjectField(obj , nameFieldId , cName); // 设置该字段的值  
  33. }  


四、在Native层操作Java层的类:回调Java方法 

    Java层原型方法:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     //Native层回调的方法实现  
  4.     public void callback(String fromNative){       
  5.         System.out.println(" I was invoked by native method  ############# " + fromNative);  
  6.     };  
  7.     public native void doCallBack(); //Native层会调用callback()方法  
  8.     ...   
  9.       
  10.     // main函数  
  11.     public static void main(String[] args)   
  12.     {  
  13.         new HelloJni().ddoCallBack();  
  14.     }     
  15. }     

    Native层该方法实现为 :

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    doCallBack 
  4.  * Signature: ()V 
  5.  */  
  6. //Native层回调Java类方法  
  7. JNIEXPORT void JNICALL Java_com_feixun_jni_HelloJni_doCallBack  
  8.   (JNIEnv * env , jobject obj)  
  9. {  
  10.      //回调Java中的方法  
  11.   
  12.     jclass cls = env->GetObjectClass(obj);//获得Java类实例  
  13.     jmethodID callbackID = env->GetMethodID(cls , "callback" , "(Ljava/lang/String;)V") ;//或得该回调方法句柄  
  14.   
  15.     if(callbackID == NULL)  
  16.     {  
  17.          cout << "getMethodId is failed \n" << endl ;  
  18.     }  
  19.     
  20.     jstring native_desc = env->NewStringUTF(" I am Native");  
  21.   
  22.     env->CallVoidMethod(obj , callbackID , native_desc); //回调该方法,并且传递参数值  
  23. }  


    接下来,我们会操作复杂对象,也就是Java层的类,包括从Native层返回一个类以及传递一个类到Native层去, 这儿我们

使用的类非常简单,如下:

     Student.java

[java]  view plain copy
print ?
  1. package com.feixun.jni;  
  2.   
  3. public class Student  
  4. {  
  5.     private int age ;  
  6.     private String name ;  
  7.     //构造函数,什么都不做  
  8.     public Student(){ }  
  9.       
  10.     public Student(int age ,String name){  
  11.         this.age = age ;  
  12.         this.name = name ;  
  13.     }  
  14.       
  15.     public int getAge() {  
  16.         return age;  
  17.     }  
  18.     public void setAge(int age) {  
  19.         this.age = age;  
  20.     }  
  21.     public String getName() {  
  22.         return name;  
  23.     }  
  24.     public void setName(String name){  
  25.         this.name = name;  
  26.     }  
  27.       
  28.     public String toString(){  
  29.         return "name --- >" + name + "  age --->" + age ;  
  30.     }  
  31. }  

 五、在Native层返回一个复杂对象(即一个类咯)


     Java层的方法对应为:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     //在Native层返回一个Student对象  
  4.     public native Student nativeGetStudentInfo() ;  
  5.     ...   
  6. }     

     Native层该方法实现为 :        

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    nativeGetStudentInfo 
  4.  * Signature: ()Lcom/feixun/jni/Student; 
  5.  */  
  6. //返回一个复杂对象  
  7. JNIEXPORT jobject JNICALL Java_com_feixun_jni_HelloJni_nativeGetStudentInfo  
  8.   (JNIEnv * env, jobject obl)  
  9. {  
  10.     //关于包描述符,这儿可以是 com/feixun/jni/Student 或者是 Lcom/feixun/jni/Student;   
  11.     //   这两种类型 都可以获得class引用  
  12.     jclass stucls = env->FindClass("com/feixun/jni/Student"); //或得Student类引用  
  13.   
  14.     //获得得该类型的构造函数  函数名为 <init> 返回类型必须为 void 即 V  
  15.     jmethodID constrocMID = env->GetMethodID(stucls,"<init>","(ILjava/lang/String;)V");  
  16.   
  17.     jstring str = env->NewStringUTF(" come from Native ");  
  18.   
  19.     jobject stu_ojb = env->NewObject(stucls,constrocMID,11,str);  //构造一个对象,调用该类的构造函数,并且传递参数  
  20.   
  21.   
  22.     return stu_ojb ;  
  23. }  


 六、从Java层传递复杂对象至Native层


     Java层的方法对应为:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     //在Native层打印Student的信息  
  4.     public native void  printStuInfoAtNative(Student stu);  
  5.     ...   
  6. }  

     Native层该方法实现为 :       

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    printStuInfoAtNative 
  4.  * Signature: (Lcom/feixun/jni/Student;)V 
  5.  */  
  6. //在Native层输出Student的信息  
  7. JNIEXPORT void JNICALL Java_com_feixun_jni_HelloJni_printStuInfoAtNative  
  8.   (JNIEnv * env, jobject obj,  jobject obj_stu) //第二个类实例引用代表Student类,即我们传递下来的对象  
  9. {  
  10.       
  11.     jclass stu_cls = env->GetObjectClass(obj_stu); //或得Student类引用  
  12.   
  13.     if(stu_cls == NULL)  
  14.     {  
  15.         cout << "GetObjectClass failed \n" ;  
  16.     }  
  17.     //下面这些函数操作,我们都见过的。O(∩_∩)O~  
  18.     jfieldID ageFieldID = env->GetFieldID(stucls,"age","I"); //获得得Student类的属性id   
  19.     jfieldID nameFieldID = env->GetFieldID(stucls,"name","Ljava/lang/String;"); // 获得属性ID  
  20.   
  21.     jint age = env->GetIntField(objstu , ageFieldID);  //获得属性值  
  22.     jstring name = (jstring)env->GetObjectField(objstu , nameFieldID);//获得属性值  
  23.   
  24.     const char * c_name = env->GetStringUTFChars(name ,NULL);//转换成 char *  
  25.    
  26.     string str_name = c_name ;   
  27.     env->ReleaseStringUTFChars(name,c_name); //释放引用  
  28.       
  29.     cout << " at Native age is :" << age << " # name is " << str_name << endl ;   
  30. }  


 七、最后加个难度,即在Native层返回集合对象(留这儿,以后也好找点)


     Java层的对应方法为:

[java]  view plain copy
print ?
  1. public class HelloJni {  
  2.     ...  
  3.     //在Native层返回ArrayList集合   
  4.     public native ArrayList<Student> native_getListStudents();  
  5.     ...   
  6. }     

     Native层该方法实现为 :        

[java]  view plain copy
print ?
  1. /* 
  2.  * Class:     com_feixun_jni_HelloJni 
  3.  * Method:    native_getListStudents 
  4.  * Signature: ()Ljava/util/ArrayList; 
  5.  */ //获得集合类型的数组  
  6. JNIEXPORT jobject JNICALL Java_com_feixun_jni_HelloJni_native_getListStudents  
  7.   (JNIEnv * env, jobject obj)  
  8. {  
  9.     jclass list_cls = env->FindClass("Ljava/util/ArrayList;");//获得ArrayList类引用  
  10.   
  11.     if(listcls == NULL)  
  12.     {  
  13.         cout << "listcls is null \n" ;  
  14.     }  
  15.     jmethodID list_costruct = env->GetMethodID(list_cls , "<init>","()V"); //获得得构造函数Id  
  16.   
  17.     jobject list_obj = env->NewObject(list_cls , list_costruct); //创建一个Arraylist集合对象  
  18.     //或得Arraylist类中的 add()方法ID,其方法原型为: boolean add(Object object) ;  
  19.     jmethodID list_add  = env->GetMethodID(list_cls,"add","(Ljava/lang/Object;)Z");   
  20.     
  21.     jclass stu_cls = env->FindClass("Lcom/feixun/jni/Student;");//获得Student类引用  
  22.     //获得该类型的构造函数  函数名为 <init> 返回类型必须为 void 即 V  
  23.     jmethodID stu_costruct = env->GetMethodID(stu_cls , "<init>""(ILjava/lang/String;)V");  
  24.   
  25.     for(int i = 0 ; i < 3 ; i++)  
  26.     {  
  27.         jstring str = env->NewStringUTF("Native");  
  28.         //通过调用该对象的构造函数来new 一个 Student实例  
  29.         jobject stu_obj = env->NewObject(stucls , stu_costruct , 10,str);  //构造一个对象  
  30.           
  31.         env->CallBooleanMethod(list_obj , list_add , stu_obj); //执行Arraylist类实例的add方法,添加一个stu对象  
  32.     }  
  33.   
  34.     return list_obj ;  
  35. }  



         最后,如何调用这些JNI函数,大家都懂的,直接调用即可,我就不在贴代码了,免得罗嗦。



        OK,本次JNI的学习就算告一段落了。下一步该认真仔细学习下Android中JNI的使用了。哎,怎么学的东西又那么多呢? - -

这篇关于http://blog.csdn.net/qinjuning/article/details/7607214的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/994436

相关文章

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

Python如何实现 HTTP echo 服务器

《Python如何实现HTTPecho服务器》本文介绍了如何使用Python实现一个简单的HTTPecho服务器,该服务器支持GET和POST请求,并返回JSON格式的响应,GET请求返回请求路... 一个用来做测试的简单的 HTTP echo 服务器。from http.server import HT

.NET利用C#字节流动态操作Excel文件

《.NET利用C#字节流动态操作Excel文件》在.NET开发中,通过字节流动态操作Excel文件提供了一种高效且灵活的方式处理数据,本文将演示如何在.NET平台使用C#通过字节流创建,读取,编辑及保... 目录用C#创建并保存Excel工作簿为字节流用C#通过字节流直接读取Excel文件数据用C#通过字节

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

如何在Visual Studio中调试.NET源码

今天偶然在看别人代码时,发现在他的代码里使用了Any判断List<T>是否为空。 我一般的做法是先判断是否为null,再判断Count。 看了一下Count的源码如下: 1 [__DynamicallyInvokable]2 public int Count3 {4 [__DynamicallyInvokable]5 get

2、PF-Net点云补全

2、PF-Net 点云补全 PF-Net论文链接:PF-Net PF-Net (Point Fractal Network for 3D Point Cloud Completion)是一种专门为三维点云补全设计的深度学习模型。点云补全实际上和图片补全是一个逻辑,都是采用GAN模型的思想来进行补全,在图片补全中,将部分像素点删除并且标记,然后卷积特征提取预测、判别器判别,来训练模型,生成的像

BUUCTF靶场[web][极客大挑战 2019]Http、[HCTF 2018]admin

目录   [web][极客大挑战 2019]Http 考点:Referer协议、UA协议、X-Forwarded-For协议 [web][HCTF 2018]admin 考点:弱密码字典爆破 四种方法:   [web][极客大挑战 2019]Http 考点:Referer协议、UA协议、X-Forwarded-For协议 访问环境 老规矩,我们先查看源代码

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而