Android JNI 入门(含完整Demo),Android开发社招面试经验

第一步:进入到项目中的 java 目录

我的地址为:
cd Desktop/AndroidProject/JNI/app/src/main/java/

可以通过 Reveal in Finder 快速打开当前文件

Android JNI 入门(含完整Demo),Android开发社招面试经验\

第二步:输入java指令

javah com.example.jni.MainActivity
(javah 类名.包名) 

然后就看到了生成的.h文件

Android JNI 入门(含完整Demo),Android开发社招面试经验\

MainActivity中定义的常量,自动生成的宏!

Android JNI 入门(含完整Demo),Android开发社招面试经验

.h文件内容:

//JNI 头文件
#include <jni.h>

//解决循环拷贝问题
#ifndef _Included_com_example_jni_MainActivity //如果没有定义这个宏
#define _Included_com_example_jni_MainActivity//就定义这个宏

#ifdef __cplusplus //如果是 C++环境,就使用 C 的环境
extern "C" {//全部采用 C 的方式,采用 C 是为了不准你函数重载(解决函数名冲突的问题)
#endif
#undef com_example_jni_MainActivity_A

//自动检测 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)
     *      :情况二:(jclass)静态: 谁调用它,就是谁的class,这里 MainActivity 调用,jclass 就是 MainActivity.class
     */
} 

参数一: 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 JNICALL
Java_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()修改为’李四’

属性签名最后会说!

使用:

Android JNI 入门(含完整Demo),Android开发社招面试经验

运行结果为:

Android JNI 入门(含完整Demo),Android开发社招面试经验

[](

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

native 层改变静态属性:

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

vative 层代码:

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

结尾

最后小编想说:不论以后选择什么方向发展,目前重要的是把Android方面的技术学好,毕竟其实对于程序员来说,要学习的知识内容、技术有太多太多,要想不被环境淘汰就只有不断提升自己,从来都是我们去适应环境,而不是环境来适应我们!

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

想要拿高薪实现技术提升薪水得到质的飞跃。最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以为了大家能够顺利进阶中高级、架构师,我特地为大家准备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。

**[CodeChina开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》](

)**

当你有了学习线路,学习哪些内容,也知道以后的路怎么走了,理论看多了总要实践的。

Android JNI 入门(含完整Demo),Android开发社招面试经验

高级UI,自定义View

UI这块知识是现今使用者最多的。当年火爆一时的Android入门培训,学会这小块知识就能随便找到不错的工作了。

不过很显然现在远远不够了,拒绝无休止的CV,亲自去项目实战,读源码,研究原理吧!

Android JNI 入门(含完整Demo),Android开发社招面试经验

容,也知道以后的路怎么走了,理论看多了总要实践的。

[外链图片转存中…(img-ry8a35zp-1631339395211)]

高级UI,自定义View

UI这块知识是现今使用者最多的。当年火爆一时的Android入门培训,学会这小块知识就能随便找到不错的工作了。

不过很显然现在远远不够了,拒绝无休止的CV,亲自去项目实战,读源码,研究原理吧!

[外链图片转存中…(img-W9naCDYh-1631339395211)]

上一篇:数据绑定控件之Repeater


下一篇:为何触发此事件?