JNI的簡單使用網(wǎng)上有很多,但大多數(shù)都是基于Android.mk的方式,但是新的NDK都是使用CmakeList的方式配置,新建一個支持c++的android 項目也是如此,所以主要還是要使用CmakeList的方式配置。
1、JNI環(huán)境搭建
ndk下載:https://developer.android.google.cn/ndk/downloads/
ndk配置:


然后創(chuàng)建一個支持C++的Android項目,就大功告成 就這么簡單!
什么?你不會創(chuàng)建支持C++的Android項目?你咋不上天呢?那你還玩啥?。?br>
CmakeList簡單語法介紹:
https://jingyan.baidu.com/album/414eccf6a577946b431f0a05.html?picindex=1
建議先學習一下CmakeList
2、java調(diào)用C++代碼
一般我們會按照項目給的方式去使用,頂多是我們多加幾個C文件,添加完重新編譯,就會自己打包成新的so文件,c都是其它同事完成的我只負責使用JNI調(diào)用,但是最好還是會C語言 C++。
幾個需要注意點:
確定按照CmakeList來編譯程序
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
創(chuàng)建Native方法,可以在Activity也可以新建一個類都可以:
public static native int ToCPP();
public static native void ToJava();

c++程序代碼,要與java類對應
JNIEXPORT jint JNICALL
Java_com_topotek_jnitest_Commond_ToCPP(
JNIEnv *env,
jobject /* this */) {
return 1;
}
添加C++代碼到Android項目

最后在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調(diào)用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去調(diào)用C++
public static native void onPreviewFrame(byte[] frameData, int frameWidth, int frameHeight);
public static native void onCommand(String command);
//供給C++語言調(diào)用java
public static void getOneFrame(){
ProjectModule.getOneFrame(false);
}
public static void resultToJava(int result){
Log.i("LogWgd", "getResult: "+result);
}
}
注意包名,然后去添加C++代碼,也可以是C代碼:
JNI.h 這個是Java直接調(diào)用到這里,只聲明函數(shù),具體的實現(xiàn),在下面:
#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調(diào)用
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++調(diào)用JAVA方法,可以是靜態(tài)方法也可以是非靜態(tài)方法
1、在java中定義供給C調(diào)用的方法
package com.topotek.jnitest;
import android.util.Log;
public class Commond {
//以下供給C語言調(diào)用
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中調(diào)用,注釋的內(nèi)容可單個打開測試對應不同的java方法,
執(zhí)行流程是,先由java調(diào)用到C中的Java_com_topotek_jnitest_Commond_ToJava
方法,然后在這個C的方法中再調(diào)用Java中的方法,注釋可打開一 個 個測試,對照著寫即可?。。。?!
#include <jni.h>
#include <string>
extern "C" {
//JNI調(diào)用Java的靜態(tài)方法
//以下是關于相關知識的技術博客
//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調(diào)用非靜態(tài)方法
/* jclass cls = (*env).FindClass("com/topotek/jnitest/Commond");
jmethodID mid=(env)->GetMethodID(cls,"commond4","(I)V");
//獲取構造函數(shù)
//jmethodID id = env->GetMethodID(cls, "", "()V");
//調(diào)用構造函數(shù)創(chuàng)建對象
//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;
}
}
這里面需要注意一點就是在調(diào)用java的方法時,GetStaticMethodID 或者 是 GetMethodID 最后一個參數(shù) 叫做java方法的簽名;
生成方法如下:
//https://www.cnblogs.com/Anita9002/p/5942965.html
//生成java方法簽名 javap -s MainActivity.class
但是有人還是找不到.class在哪,編譯運行項目,然后cd到類似于這樣的目錄下
執(zhí)行命令:
//\app\build\intermediates\classes\debug\com\topotek\jnitest>javap -s Commond
在哪執(zhí)行?

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