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对象
实现效果:java 属性值为"张三",通过调用 native 函数,修改为李四
public String name = “张三”;
在 MainActivity 中创建调用 native 的方法
//通过 native 修改名字 为"李四"
public native void changeName();
native 层代码:
我直接在实现文件写了!
//NDK 工具链中的 log库(用来打印)
#include
//定义宏,用来打印结果
#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 j_fid = env->GetFieldID(j_cls, “name”, “Ljava/lang/String;”);
jstring j_str = static_cast(env->GetObjectField(thiz, j_fid));
const char *chars = env->GetStringUTFChars(j_str, NULL);
LOGD(“native %s”, chars);
LOGI(“native %s”, chars);
LOGE(“native %s”, chars);
jstring st = env->NewStringUTF(“李四”);
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 修改年龄+10
public static native void changeAge();
vative 层代码:
extern “C”
JNIEXPORT void JNICALL
Java_com_example_jni_MainActivity_changeAge(JNIEnv env, jclass clazz) {
/*
- 参数三:基本类型签名(int 对应 I)
*/
jfieldID j_id = env->GetStaticFieldID(clazz, “age”, “I”);
jint j_age = env->GetStaticIntField(clazz, j_id);
//修改参数
j_age += 10;
env->SetStaticIntField(clazz, j_id, j_age);
}
使用:
运行效果:
java层:
public int add(int number1,int number2){
return number1+ number2;
}
//native 调用 java 方法
public native int nativeAdd();
native 层:
extern “C”
JNIEXPORT jint JNICALL
Java_com_example_jni_MainActivity_nativeAdd(JNIEnv env, jobject thiz) {
/*
- 通过 jobject 获取 jclass
- jclass GetObjectClass(jobject obj)
*/
jclass j_c = env->GetObjectClass(thiz);
jmethodID jmethodId = env->GetMethodID(j_c, “add”, “(II)I”);
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;
}
运行结果为:
属性签名《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
【docs.qq.com/doc/DSkNLaERkbnFoS0ZF】 完整内容开源分享
八大基本类型签名:
| java | native |
|---|---|
| boolean | Z |
| short | S |
| int | I |
| byte | B |
| double | D |
| float | F |
| char | C |
| long | J |
方法举例:
| java | native |
|---|---|
| void sum(int a,int b) | (II)V |
| void sum(int a,int b,int c) | (III)V |
数组举例:
| java | native |
|---|---|
| array int[] | [ I |
| array int[][] | [ [ I |
| array float[][][] | [ [ [ F |
也可以通过指令来获取一个类中的所有签名:
第一步:进入到.class 目录:
进入到这个目录下:
我的地址为:
cd /Users/shizhenjiang/Desktop/AndroidProject/JNI/app/build/intermediates/javac/debug/classes/com/example/jni
然后输入指令:
javap -s -p xxx.class
| [ [ [ F |
也可以通过指令来获取一个类中的所有签名:
第一步:进入到.class 目录:
进入到这个目录下:
[外链图片转存中…(img-KOikcSLv-1639188829704)]
我的地址为:
cd /Users/shizhenjiang/Desktop/AndroidProject/JNI/app/build/intermediates/javac/debug/classes/com/example/jni
然后输入指令:
javap -s -p xxx.class



