写点什么

Android JNI 入门 (含完整 Demo)

  • 2021 年 11 月 06 日
  • 本文字数:2588 字

    阅读完需:约 8 分钟

//自动检测 MainActivity 生成宏 #define com_example_jni_MainActivity_A 234L


//函数的声明 extern "C" JNIEXPORT jstring JNICALL Java_com_example_jni_MainActivity_getString(JNIEnv *, jobject);


#ifdef __cplusplus//如果是 C++ 什么事情都不干}#endif#endif


这里注释添加的非常清楚,就不在啰嗦了,如若不懂,评论区留言哦!


函数实现文件:


//extern "C" JNIEXPORT jstring JNICALL Java_com_example_jni_MainActivity_getString// (JNIEnv *, jobject){}//=================================请看如下注释=================================


//JNIEnv 是 C++ 的最终会调用到 C 的 JNINativeInterface 方法 ,所以必须采用 C 的方式(extern "C")extern "C" //采用 C 的编译方式


JNIEXPORT //标记该方法会被外部调用(VS 会报错,AS 不会报错)


jstring // java 中方法的返回值,// 这里返回 jstring,表示 java 中是 String 类型//如果是 jint 则表示 java 中是 int 类型


JNICALL//表示是 JIN 的标记(这个可以去掉)


//函数名,由 JDK 设计的(JNI 是 java 的技术,不是 native 的技术)Java_com_example_jni_MainActivity_getString(JNIEnv env, jobject job) {/*


  • 参数一:(JNIEnv):是 Java 与 C/C++通信最重要的东西(精华)

  • 参数二 :情况一(jobject)非静态:谁调用它,就是谁的实例,这里 MainActivity 调用,job 就是 MainActivity(this)


*/}


参数一: JNIEnv 这个参数非常重要,是 JNI 的精华,这个参数最终会调用到 C 的结构体(JNINativeInterface)


这里需要注意的就是第二个参数:


  • 当为非静态的时候,生成的是 jobject 对象

  • 当为静态的时候,生成的是 jclass 对象

native 层改变 java 属性的值(非静态)

实现效果:java 属性值为"张三",通过调用 native 函数,修改为李四


public String name = "张三";


在 MainActivity 中创建调用 native 的方法


//通过 native 修改名字 为"李四"public native void changeName();


native 层代码:


我直接在实现文件写了!


//NDK 工具链中的 log 库(用来打印)#include <android/log.h>


//定义宏,用来打印结果 #define TAG "szj"//...我都不知道要传什么,可以借助 JNI 中的宏来传入 #define LOGD(...)__android_log_print(ANDROID_LOG_DEBUG,TAG,VA_ARGS);#define LOGI(...)__android_log_print(ANDROID_LOG_INFO,TAG,VA_ARGS);#define LOGE(...)__android_log_print(ANDROID_LOG_ERROR,TAG,VA_ARGS);


//函数具体实现 extern "C"JNIEXPORT void JNICALLJava_com_example_jni_MainActivity_changeName(JNIEnv env, jobject thiz) {/*


  • 获取 class*/jclass j_cls = env->GetObjectClass(thiz);


/**


  • jfieldID GetFieldID(jclass clazz, const char* name, const char* sig)

  • 参数一:MainActivity.class

  • 参数二:属性名

  • 参数三:属性签名 L 表示引用类型*/jfieldID j_fid = env->GetFieldID(j_cls, "name", "Ljava/lang/String;");


/**


  • jobject GetObjectField(jobject obj, jfieldID fieldID)

  • 把 java 属性变成 jstring 类型

  • 参数一:jobject

  • 参数二:签名 ID

  • static_cast<jstring>强制转换为 jstring 类型*/jstring j_str = static_cast<jstring>(env->GetObjectField(thiz, j_fid));


/**


  • 将 JNI 的 jstring 转换为 C++的 char 类型

  • const char* GetStringUTFChars(jstring string, jboolean* isCopy)

  • 打印字符串

  • 参数一:需要转换的字符串

  • 参数二:不知道是啥*/const char *chars = env->GetStringUTFChars(j_str, NULL);LOGD("native %s", chars);LOGI("native %s", chars);LOGE("native %s", chars);


jstring st = env->NewStringUTF("李四");


/**


  • void SetObjectField(jobject obj, jfieldID fieldID, jobject value)

  • 修改成 李四

  • 参数一:jobject

  • 参数二:签名 ID

  • 参数三:修改后的 jstring*/env->SetObjectField(thiz, j_fid, st);}


和 java 反射有异曲同工之妙;


思路:


  • 通过 GetObjectClass()获取 jclass (主要以 j 开头的都是 JNI 的属性 例如:jclass,jstring,jfieldID 等)

  • 通过 GetFieldID() 获取到属性名的 ID

  • 然后通过 GetObjectField()将 java 上的 String 改变为 JNI 认识的 jstring

  • 最后修改通过 SetObjectField()修改为’李四’


属性签名最后会说!


使用:



运行结果为:


native 层改变 java 属性的值(静态)

native 层改变静态属性:


//通过 native 修改年龄+10public static native void changeAge();


vative 层代码:


extern "C"JNIEXPORT void JNICALLJava_com_example_jni_MainActivity_changeAge(JNIEnv env, jclass clazz) {/*


  • 参数三:基本类型签名(int 对应 I)*/jfieldID j_id = env->GetStaticFieldID(clazz, "age", "I");


/**


  • 获取静态 int 类型的属性*jint GetStaticIntField(jclass clazz, jfieldID fieldID)

  • 参数一:jclass

  • 参数二:静态 属性 ID*/jint j_age = env->GetStaticIntField(clazz, j_id);


//修改参数 j_age += 10;


/**


  • 修改值后,在设置回去新的值

  • void SetStaticIntField(jclass clazz, jfieldID fieldID, jint


《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
浏览器打开:qq.cn.hn/FTe 免费领取
复制代码


value)


  • 参数一:jclass

  • 参数二:静态属性 ID

  • 参数三:新的静态值*/env->SetStaticIntField(clazz, j_id, j_age);}


使用:



运行效果:


native 调用 java 静态方法

java 层:


public int add(int number1,int number2){return number1+ number2;}//native 调用 java 方法 public native int nativeAdd();


native 层:


extern "C"JNIEXPORT jint JNICALLJava_com_example_jni_MainActivity_nativeAdd(JNIEnv env, jobject thiz) {/*


  • 通过 jobject 获取 jclass

  • jclass GetObjectClass(jobject obj)*/jclass j_c = env->GetObjectClass(thiz);


/**


  • 获取方法 ID;

  • jmethodID GetMethodID(jclass clazz, const char* name, const char* sig)

  • 参数一:jclass 对象

  • 参数二:调用的方法名

  • 参数三:参数与返回值签名*/jmethodID jmethodId = env->GetMethodID(j_c, "add", "(II)I");


/**


  • jint(CallIntMethod)(JNIEnv, jobject, jmethodID, ...);

  • 调用 java 的方法返回值为 int*/jint jint1 = env->CallIntMethod(thiz, jmethodId, 4, 2);LOGE("native %d", jint1);


//方法结束必须返回 0 ,否则会报以下错误,(不要问,问就是找了 20 分钟)//Fatal signal 5 (SIGTRAP), code 1 (TRAP_BRKPT), fault addr 0x7675a751dc in tid 18347 (com.example.jni),// pid 18347 (com.example.jni)return 0;}


运行结果为:


属性签名

八大基本类型签名:



方法举例:



数组举例:


| java | native |

评论

发布
暂无评论
Android JNI 入门(含完整Demo)