美文网首页
Android中 JNI的使用:java调用C++ C++调用J

Android中 JNI的使用:java调用C++ C++调用J

作者: 从新开始学android | 来源:发表于2018-07-26 17:02 被阅读0次

    JNI的简单使用网上有很多,但大多数都是基于Android.mk的方式,但是新的NDK都是使用CmakeList的方式配置,新建一个支持c++的android 项目也是如此,所以主要还是要使用CmakeList的方式配置。

    1、JNI环境搭建

    ndk下载:https://developer.android.google.cn/ndk/downloads/
    ndk配置:

    ndk配置1 ndk配置2

    然后创建一个支持C++的Android项目,就大功告成 就这么简单!
    什么?你不会创建支持C++的Android项目?你咋不上天呢?那你还玩啥!!
    CmakeList简单语法介绍:
    https://jingyan.baidu.com/album/414eccf6a577946b431f0a05.html?picindex=1
    建议先学习一下CmakeList

    2、java调用C++代码

    一般我们会按照项目给的方式去使用,顶多是我们多加几个C文件,添加完重新编译,就会自己打包成新的so文件,c都是其它同事完成的我只负责使用JNI调用,但是最好还是会C语言 C++。
    几个需要注意点:
    确定按照CmakeList来编译程序

     externalNativeBuild {
            cmake {
                path "CMakeLists.txt"
            }
        }
    

    创建Native方法,可以在Activity也可以新建一个类都可以:

        public static native int ToCPP();
    
        public static native void ToJava();
    
    native方法的位置.png

    c++程序代码,要与java类对应

    JNIEXPORT jint JNICALL
    Java_com_topotek_jnitest_Commond_ToCPP(
            JNIEnv *env,
            jobject /* this */) {
        return 1;
    }
    

    添加C++代码到Android项目


    cmakeList里面添加新的cpp文件.png

    最后在Activity中引入 Library

        /**
         * A native method that is implemented by the 'native-lib' native library,
         * which is packaged with this application.
         */
        public native String stringFromJNI();
        // Used to load the 'native-lib' library on application startup.
        static {
            System.loadLibrary("native-lib");
        }
    

    然后在任意位置 都可以 java调用c++:

     int a=Commond.ToCPP();
    

    下面举例 封装一个 更具体实用的例子:

    package com.topotek.movidius.jni;
    
    import android.util.Log;
    
    import com.topotek.topotekmodule.ProjectModule;
    /**
     * Created by wgd on 2018/5/18.
     * C 方法
     */
    public class JNI {
        
        //java去调用C++
    
        public static native void onPreviewFrame(byte[] frameData, int frameWidth, int frameHeight);
    
        public static native void onCommand(String command);
    
        //供给C++语言调用java
        public static void getOneFrame(){
            ProjectModule.getOneFrame(false);
        }
        public static void resultToJava(int result){
            Log.i("LogWgd", "getResult: "+result);
        }
        
    }
    

    注意包名,然后去添加C++代码,也可以是C代码:
    JNI.h 这个是Java直接调用到这里,只声明函数,具体的实现,在下面:

    #include <jni.h>
    //
    // Created by topptek on 2018/5/18.
    //
    #ifndef ADDLIB_COM_TOPOTEK_MOVIDIUS_JNI_JNI_H
    #define ADDLIB_COM_TOPOTEK_MOVIDIUS_JNI_JNI_H
    #endif //ADDLIB_COM_TOPOTEK_MOVIDIUS_JNI_JNI_H
    #ifdef __cplusplus
    extern "C" {
    #endif
    
    JNIEXPORT void JNICALL Java_com_topotek_movidius_jni_JNI_onPreviewFrame
            (JNIEnv *, jclass, jbyteArray, jint, jint);
    
    JNIEXPORT void JNICALL Java_com_topotek_movidius_jni_JNI_onCommand
            (JNIEnv *, jclass, jstring);
    
    #ifdef __cplusplus
    }
    #endif
    
    

    callback.h

    #include "com_topotek_movidius_jni_JNI.h"
    //
    // Created by topptek on 2018/5/18.
    //
    #ifndef ADDLIB_CALLBACK_H
    #define ADDLIB_CALLBACK_H
    #endif //ADDLIB_CALLBACK_H
    
    //给java调用
    void onPreviewFrame(JNIEnv *, signed char *, int, int, int);
    
    void onCommand(JNIEnv *, char *);
    
    void Java_com_topotek_movidius_jni_JNI_onPreviewFrame(JNIEnv *jniEnv0, jclass jclass0, jbyteArray frameData, jint frameWidth, jint frameHeight) {
    //    jsize length = (*jniEnv0)->GetArrayLength(jniEnv0, frameData);
        jsize length = jniEnv0->GetArrayLength(frameData);
    //    jbyte *byteArrayElements = (*jniEnv0)->GetByteArrayElements(jniEnv0, frameData, 0);
        jbyte *byteArrayElements = jniEnv0->GetByteArrayElements(frameData, 0);
        onPreviewFrame(jniEnv0, byteArrayElements, length, frameWidth, frameHeight);
    //    (*jniEnv0)->ReleaseByteArrayElements(jniEnv0, frameData, byteArrayElements, 0);
        jniEnv0->ReleaseByteArrayElements(frameData, byteArrayElements, 0);
    }
    
    void Java_com_topotek_movidius_jni_JNI_onCommand(JNIEnv *jniEnv0, jclass jclass0, jstring command) {
    //    const char *string = (*jniEnv0)->GetStringUTFChars(jniEnv0, command, 0);
        const char *string = jniEnv0->GetStringUTFChars(command, 0);
        onCommand(jniEnv0, (char *) string);
    //    (*jniEnv0)->ReleaseStringUTFChars(jniEnv0, command, string);
        jniEnv0->ReleaseStringUTFChars(command, string);
    }
    

    然后在C或c++中,只要引入callback.h就可以使用 onPreviewFrame和OnCommond方法了。

    JNIDemo1(addLib):完成了Movidius的移植 也可以光参考JNI部分代码,这个代码你不一定能运行起来因为需要连接Movidius才行:
    https://github.com/wangguodonggood/AddLib

    3、C++调用JAVA方法,可以是静态方法也可以是非静态方法

    1、在java中定义供给C调用的方法

    package com.topotek.jnitest;
    import android.util.Log;
    public class Commond {
        //以下供给C语言调用
        public static void commond1(float a) {
            Log.i("TTTTT", "commond1: ");
            Log.i(TAG, "commodTest: ---------" + a);
        }
    
        public static void commond2(String s) {
            Log.i(TAG, "commodTest: ---------" + s);
        }
    
        public static void commond3(int a, String s) {
            Log.i(TAG, "commodTest: ----S:" + s + "--A:" + a);
        }
    
        public void commond4(int a) {
            Log.i(TAG, "commodTest: ------A:" + a);
        }
    
    }
    

    2、在C中调用,注释的内容可单个打开测试对应不同的java方法,
    执行流程是,先由java调用到C中的Java_com_topotek_jnitest_Commond_ToJava
    方法,然后在这个C的方法中再调用Java中的方法,注释可打开一 个 个测试,对照着写即可!!!!!

    #include <jni.h>
    #include <string>
    extern "C" {
    //JNI调用Java的静态方法
    //以下是关于相关知识的技术博客
    //https://www.cnblogs.com/xitang/p/4174619.html
    //https://www.cnblogs.com/Anita9002/p/5942965.html
    //http://hubingforever.blog.163.com/blog/static/171040579201151722833782/
    //https://blog.csdn.net/qq_27278957/article/details/77164353
    //如果声明了指针记得释放指针
    void Java_com_topotek_jnitest_Commond_ToJava(
            JNIEnv *env,
            jobject obj) {
        jclass cls = (*env).FindClass("com/topotek/jnitest/Commond");
        jmethodID mid = (env)->GetStaticMethodID(cls, "commond1", "(F)V");
        (env)->CallStaticVoidMethod(cls, mid, 0.666);
    
      /*  const char* ss="这是从jni传来的字符串";
        jclass cls = (*env).FindClass("com/topotek/jnitest/Commond");
        jmethodID mid = (env)->GetStaticMethodID(cls, "commond2", "(Ljava/lang/String;)V");
        (env)->CallStaticVoidMethod(cls, mid, env->NewStringUTF(ss));*/
    
    /*    const char* ss="这是从jni传来的字符串";
        jclass cls = (*env).FindClass("com/topotek/jnitest/Commond");
        jmethodID mid = (env)->GetStaticMethodID(cls, "commond3", "(ILjava/lang/String;)V");
        (env)->CallStaticVoidMethod(cls, mid,666,env->NewStringUTF(ss));*/
    
        //JNI调用非静态方法
     /*   jclass cls = (*env).FindClass("com/topotek/jnitest/Commond");
        jmethodID  mid=(env)->GetMethodID(cls,"commond4","(I)V");
        //获取构造函数
        //jmethodID id = env->GetMethodID(cls, "", "()V");
        //调用构造函数创建对象
        //jobject j=(env)->NewObject(cls,id);
        jobject  jobject1=env->AllocObject(cls);
        (env)->CallVoidMethod(jobject1,mid,1000);*/
    
    }
    JNIEXPORT jint JNICALL
    Java_com_topotek_jnitest_Commond_ToCPP(
            JNIEnv *env,
            jobject /* this */) {
        return 1;
    }
    }
    

    这里面需要注意一点就是在调用java的方法时,GetStaticMethodID 或者 是 GetMethodID 最后一个参数 叫做java方法的签名;
    生成方法如下:
    //https://www.cnblogs.com/Anita9002/p/5942965.html
    //生成java方法签名 javap -s MainActivity.class
    但是有人还是找不到.class在哪,编译运行项目,然后cd到类似于这样的目录下
    执行命令:
    //\app\build\intermediates\classes\debug\com\topotek\jnitest>javap -s Commond
    在哪执行?

    生成java方法签名.png

    以下是完整demo:(JNITest):
    https://github.com/wangguodonggood/JNITest/tree/master

    相关文章

      网友评论

          本文标题:Android中 JNI的使用:java调用C++ C++调用J

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