在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
java与c/c++之间的数据交互
JNI 上述两篇文章对jni的讲解比较详细,各有利弊,就文章1来说,开门见山,直接阐述了java和C/C++的交互方式;文章2是一篇百度文库中的内容,则讲的比较详细,从各个方面阐述了jni到底是个怎么一回事,很适合初学者看。 JNI是Java Native Interface的缩写,它是java平台的一部分,允许Java代码和其他语言写的代码进行交互。使用java与本地已编译的代码交互,通常会丧失平台可移植性。但是,有些情况下这样做是可以接受的,甚至是必须的。JNI标准至少要保证本地代码能工作在任何Java 虚拟机环境下。 在写这篇文章的时候,虽然技术不怎么好,但却是一个实打实的C++程序员,因此在这篇文章中有关java的内容可能表述的不是那么准确。 说到jni,个人觉得有这么一句话说的比较好:只有当你必须在同一进程中调用本地代码时,再使用JNI。为什么这么说,原因很简单,文章开头已经说过了,为什么要写这么文章,原因更加明确了,项目需要,不使用不行。 废话不多说,接下来我就讲述下我个人使用jni时的一些理解和觉着比较重要的地方,在讲解jni时,我从以下方面讲解:头文件生成、类型转换、java代码通过jni调用C++和C++通过jni调用java,至于java怎么加载jni动态库我就不在文章中描述了,毕竟java不太懂。 头文件生成 编写带有native声明的方法的java类,使用javac命令编译所编写的java类,然后使用javah + java类名生成扩展名为h的头文件 public native void displayHelloWorld();//所有native关键词修饰的都是对本地的声明 JNIEXPORT void JNICALL Java_HelloWorld_displayHelloWorld(JNIEnv*,jobject);//生成的jni层头文件,在此详细解释下这个函数声明的意思,JNIEXPORT:导出函数、void:U返回值、JNICALL:jni调用接口、Java:java语言、HelloWorld:方法所述类、displayHelloWorld:方法名称、JNIEnv*指针指向一个函数指针表,在VC中可以直接用"->"操作符访问其中的函数、 jobject 指向在此 Java 代码中实例化的 Java 对象 LocalFunction的一个句柄,相当于this指针 类型转换 JNI环境变量——JNIEnv*的使用 说到类型转换,由于作者在使用jni的过程中,jni方法的参数都是json的字符串类型传递,因此网上传递数组的方式我就不多解释了,反正用的也不多 首先我说下jstring转换为char *,为了方便使用,我们可以自定义类,把这两个方法分别封装在类的构造函数和析构函数中。 const char* GetStringUTFChars (jstring string,jboolean* isCopy) 返回指向字符串UTF编码的指针,如果不能创建这个字符数组,返回null。这个指针在调用ReleaseStringUTFChar()函数之前一直有效。 参数:string Java字符串对象,isCopy 如果进行拷贝,指向以JNI_TRUE填充的jboolean,否则指向以JNI_FALSE填充的jboolean。 void ReleaseStringUTFChars(jstring str, const char* chars) 通知虚拟机本地代码不再需要通过chars访问Java字符串。 参数:string Java字符串对象,chars 由GetStringChars返回的指针 下边是char * 转换为jstring,为了方便起见还是封装成类来使用。 jstring NewStringUTF(const char *utf) 返回一个新的Java字符串并将utf内容拷贝入新串,如果不能创建字符串对象,返回null。通常在反值类型为string型时用到。 参数:utf UTF编码的字符串指针,对于数值型参数,在C/C++中可直接使用void DeleteLocalRef(jstring str)释放jstring串和本地字符串的关联关系。 java代码通过jni调用C++ java写的native方法调用时对应的就调用了jni层的方法,然后我们只需要实现jni的方法时,调用我们自己的C++嗲吗,就实现了java掉C++。 C++通过jni调用java jmethodID:保存java方法ID的数组,获取方法下文有讲解 CallVoidMethod:JNIEnv *方法,参数1:jobject对象,由jni层传入,并暂时保存在jni层、参数2:java方法ID、后边一次是被调用方法的参数 JavaVM::AttachCurrentThread:附加到当前线程 JavaVM::DetachCurrentThread:从当前线程脱离 JNIEnv *pEnv; global_vm->GetEnv((void**)&pEnv, JNI_VERSION_1_6); bool isAttached = false; if (pEnv == NULL) { int status = global_vm->AttachCurrentThread(&pEnv, NULL); if (status < 0) { return; } isAttached = true; } JniString *jsSourceID = new JniString(pEnv, strGroupID.c_str()); pEnv->CallVoidMethod(m_pCallback, m_pMethodId[15], jsSourceID->getL(), nRet); delete jsSourceID; if (isAttached) global_vm->DetachCurrentThread(); JniString:自己封装的类,把char *变量封装成jstring类型变量 jobject m_pCallback = pEnv->NewGlobalRef(pCallback);//pCallback为jObject,由java代码引入,jni层注册到此 jclass m_pClass = (jclass)pEnv->NewGlobalRef( pEnv->GetObjectClass(m_pCallback)); m_pMethodId[15] = pEnv->GetMethodID(m_pClass, "OnUpdateGroupSite", "(Ljava/lang/String;I)V"); |
2023-10-27
2022-08-15
2022-08-17
2022-09-23
2022-08-13
请发表评论