美文网首页
JNI 开发流程

JNI 开发流程

作者: heiheiwanne | 来源:发表于2016-08-20 19:39 被阅读86次

    java调用C语言

    比较好的文章:(调试)Android Studio中JNI程序的单步调试和日志打印 - shenzhigang - 博客园  

    (快捷键设置):  超级简单的Android Studio jni 实现(无需命令行) - 简书

    1.配置NDK [自行配置,当然也可以不使用NDK,利用gcc 命令进行

    gcc test_a.c test_b.c test_c.c -fPIC -shared -o libtest.so

    -fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码

    是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程

    需要,而不能达到真正代码段共享的目的。可用 nm libgdfont.so 来看

    里面导出的函数(带 T 标记)。

    用动态库的好处是:更新了动态库之后链结它的程序不用重新编译。

    (gcc test.c -L. -ltest -o test)]

    Android studio中的项目中保证这几个文件都是对的:(1)gradle.properties  :android.useDeprecatedNdk=true

    (2)local.properties:这个也可以使用file下的配置

    ndk.dir=/Users/bjhl/Library/android-ndk-r11b

    sdk.dir=/Users/bjhl/Library/Android/sdk

    2.建立一个JniTest.java 类,加载jni的.so的方法,并在里面写调用C语言的函数

    System.loadLibrary("jni-test");  加载的so文件模块为jni-test    (这里我们最终生成的为libjni-test.so,这里提前展示一下)

    3.进入此类的目录生成调用的头文件

    使用命令:javah -d  ../jni  -jni com.example.bjhl.testjni.JniTest  

    即生成头文件  com_example_bjhl_testjni_JniTest.h

    (1).  ../jni  指定生成的目录,将生成的头文件放在此目录下 

    注意Android的目录结构:jni  jnilibs 都是跟java同级的(要是不是gradle单独配置),  libs (jar包的存放),是跟src同级 ,尽量使用此标准目录

    (2).  -jni 固定命令 进行jni编译

    (3).com.example.bjhl.testjni.JniTest  JniTest.java 的全类名(Copy Reference)

    (3). 在gradle.build 下写入ndk的配置:

    ndk {

    moduleName"moduleName"

    ldLibs"log"//实现__android_log_print

    abiFilters"armeabi","armeabi-v7a","x86"

    }

    (4).在jni目录下新建一个C文件实现JNi的函数,这里可以直接粘贴com.example.bjhl.testjni.JniTest.h 的函数

    OK大功告成,build即可,安装

    打印调试日志:

    (1). gradle.build 中配置ldLibs"log"//实现__android_log_print

    (2).使用__android_log_print即可(引入头函数#include"android/log.h")

    __android_log_print(ANDROID_LOG_ERROR,"MDK","%s",string);

    C语言调用java:

    实现原理:使用JNI提供的反射借口来反射得到Java方法,进行调用。

    1.c 调用java的静态方法:

    (1)获取类Class对象

    jclass clazz = (*env)->FindClass(env,"com/example/demo/jnidemo/FooCDiaoJava");

    (2)获取java函数的id值: 命令 javap -s FooCDiaoJava  获取方法参数的签名  (其中-s后的.java对象)

    进入class文件夹

    执行命令

    jmethodID  id = (*env)->GetStaticMethodID(env,clazz,"cDiao","(Ljava/lang/String;)V");

    (3)定义参数

    jstring msg = (*env)->NewStringUTF(env,"jniCodeDiao.c calljavaMethod");

    (4)进行方法的调用

    (*env)->CallStaticVoidMethod(env,clazz,id,msg);

    2.c调用java中的非静态方法  无参也无返回值:

    3.c调用java中的非静态方法 有参 无返回值:

    4.这是c调用java语言有返回值有参数:

    5.c通过new对象的方式调用java的非静态方法:

    #include "com_example_demo_jnidemo_FooCDiaoJava.h"

    #include "jni.h"

    #include "android/log.h"

    #define LOG_TAG "System.out"

    #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)

    /**

    * c调用java静态方法成功

    *

    *命令javap -s FooCDiaoJava获取方法参数的签名(其中-s后的.java对象)

    */

    JNIEXPORTvoidJNICALL Java_com_example_demo_jnidemo_FooCDiaoJava_cDiaojavaJingTai

    (JNIEnv *env, jobject jobject1){

    //获取类Class对象

    jclass clazz = (*env)->FindClass(env,"com/example/demo/jnidemo/FooCDiaoJava");

    //获取java函数的id值

    jmethodID id = (*env)->GetStaticMethodID(env,clazz,"cDiao","(Ljava/lang/String;)V");

    if(id==0){

    LOGD("find method1 error");

    return;

    }

    //定义参数

    jstring msg = (*env)->NewStringUTF(env,"jniCodeDiao.c calljavaMethod");

    //进行方法的调用

    (*env)->CallStaticVoidMethod(env,clazz,id,msg);

    }

    /**

    * c调用java中的非静态方法无参也无返回值

    *直接通过参数中的对象

    */

    JNIEXPORTvoidJNICALL Java_com_example_demo_jnidemo_FooCDiaoJava_javaDiaoC1

    (JNIEnv *env, jobject obj){

    //获得obj对象

    jclass cla = (*env)->GetObjectClass(env,obj);

    jmethodID id = (*env)->GetMethodID(env,cla,"cDiaoJava1","()V");

    //(*env)->CallVoidMethod(env,cla,id);这种写法是错误的我把jobject对象传错了

    (*env)->CallVoidMethod(env,obj,id);

    }

    /**

    * c调用java中的非静态方法有参无返回值

    */

    JNIEXPORTvoidJNICALL Java_com_example_demo_jnidemo_FooCDiaoJava_javaDiaoC2

    (JNIEnv *env, jobject obj){

    jclass cla = (*env)->GetObjectClass(env,obj);

    jmethodID id = (*env)->GetMethodID(env,cla,"cDiaoJava2","(Ljava/lang/String;)V");

    jstring str = (*env)->NewStringUTF(env,"c的参数");

    (*env)->CallVoidMethod(env,obj,id,str);

    }

    /**

    *这是c调用java语言有返回值有参数

    */

    JNIEXPORTvoidJNICALL Java_com_example_demo_jnidemo_FooCDiaoJava_javaDiaoC3

    (JNIEnv *env, jobject obj){

    //得到传入对对象的Class对象

    jclass clazz = (*env)->GetObjectClass(env,obj);

    jmethodID methodId = (*env)->GetMethodID(env,clazz,"cDiaoJava3","(I)Ljava/lang/String;");

    jstring str = (*env)->CallObjectMethod(env,obj,methodId,2);

    //经测返回值可用

    }

    /**

    * c通过new对象的方式调用java的非静态方法

    */

    JNIEXPORTvoidJNICALL Java_com_example_demo_jnidemo_FooCDiaoJava_cNewObjectDiaoJava

    (JNIEnv *env, jobject obj){

    //C中映射类

    jclass clazz = (*env)->FindClass(env,"com/example/demo/jnidemo/FooCDiaoJava");

    //C中新建对象

    jmethodID construction_id= (*env)->GetMethodID(env,clazz,"","()V");

    jobject javaObj = (*env)->NewObject(env,clazz,construction_id);

    //C中映射非静态方法

    jmethodID foocdjanewobj = (*env)->GetMethodID(env,clazz,"cNewObjDJ","()V");

    //C中调用Java非静态的方法

    (*env)->CallVoidMethod(env,javaObj,foocdjanewobj);

    }

    ```

    hashdhsafhasfd
    ```

    相关文章

      网友评论

          本文标题:JNI 开发流程

          本文链接:https://www.haomeiwen.com/subject/xfyisttx.html