zoukankan      html  css  js  c++  java
  • jni c++

        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");
    至此,java和C++的相互调用就基本讲完了,如果有什么不对的地方或者不理解的地方,欢迎大家留言反问!!!jni <wbr>c++jni <wbr>c++

  • 相关阅读:
    LeetCode算法题-Factorial Trailing Zeroes(Java实现)
    LeetCode算法题-Excel Sheet Column Number(Java实现)
    FluentData 学习 第一弹
    加油 加油
    FluentData -Micro ORM with a fluent API that makes it simple to query a database
    RX
    创业哲学
    9种新思想
    事件总线 EventBus
    关于 C#和.net 的 发展
  • 原文地址:https://www.cnblogs.com/swarmbees/p/5621406.html
Copyright © 2011-2022 走看看