目录
前言
之前其实写过,代码不见了,为了走一下流程,重新简单的写了一个。
PS:设计NDK环境配置这里不介绍哈
正文
静态注册
先由Java得到本地方法的声明,然后再通过实现该声明方法。
优点: 理解和使用方式简单, 属于傻瓜式操作, 使用相关工具按流程操作就行, 出错率低
缺点: 当需要更改类名,包名或者方法时, 需要按照之前方法重新生成头文件, 灵活性不高
实战
定义Hello.java
加载Hello.so和native方法
package com.biumall.jni.one; public class Hello { //1. load Hello.so static { System.loadLibrary("Hello"); } // 2. define native hello() public static native String hello(); }
javac编译class
通过javac编译Hello.class,然后javah编译
进入Hello.java目录
#当前位置:BiuJniStatic\src\main\java\com\biumall\jni\one javac Hello.java
会在当前目录下生成
Hello.class
javah编译.h头文件
回退到 java目录
#当前位置:BiuJniStatic\src\main\java javah com.biumall.jni.one.Hello
会在当前目录生成
com_biumall_jni_one_Hello.h
创建JNI
工程更目录创建JNI目录
BiuJniStatic/jni
把上面com_biumall_jni_one_Hello.h头文件拷贝到jni目录,并改名为
Hello.h
Hello.h拷贝一份为Hello.c
Hello.c也可以改为Hello.cpp,cpp中需要用C++写,否则会报错。
如果Hello.c中用了C++语法就会提示
error: member reference base type 'JNIEnv' (aka 'const struct JNINativeInterface *') is not a structure or union]
反过了Hello.cpp中用C语言写也会有类似的提示。
编写Hello.c
下面代码新增log打印,新增的活修改的都有备注
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_biumall_jni_one_Hello */ //---- add start ----- #include <android/log.h> //---- add end ----- #ifndef _Included_com_biumall_jni_one_Hello #define _Included_com_biumall_jni_one_Hello #ifdef __cplusplus extern "C" { #endif /* * Class: com_biumall_jni_one_Hello * Method: hello * Signature: ()Ljava/lang/String; */ //---- modify start ----- //LOG_TAG #define LOG_TAG "from_static_jni_" #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__) #define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__) #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__) JNIEXPORT jstring JNICALL Java_com_biumall_jni_one_Hello_hello (JNIEnv * env , jclass jclazz){ // [hello.c如果用C] LOGE("Java_com_biumall_jni_one_Hello_hello()"); return (*env)->NewStringUTF(env, "Hello World !!!! --- from JNI "); // [hello.cpp如果用C++] // 如果hello.c中用了这个,无法build过,提示[error: member reference base type 'JNIEnv' (aka 'const struct JNINativeInterface *') is not a structure or union] //return env->NewStringUTF((char *)"Hello World !---from JNI"); }; //---- modify end ----- #ifdef __cplusplus } #endif #endif
PS : 上面—- 等备注的都是有更新的地方,其他的都是Hello.h一样。
配置Application.mk和Android.mk
Android.mk
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_LDLIBS := -llog LOCAL_MODULE := Hello LOCAL_SRC_FILES := Hello.c include $(BUILD_SHARED_LIBRARY)
LOCAL_MODULE : 为生成so库的名字
LOCAL_SRC_FILES : 为需要编译的源文件
Application.mk
APP_ABI := all
all表示全部平台,可编译出如下so库
├─arm64-v8a ├─armeabi ├─armeabi-v7a ├─mips ├─mips64 ├─x86 └─x86_64
当然,可以指定编译指定平台的so。
# 指定生成armeabi-v7a APP_ABI := armeabi-v7a # 或者指定多个[多个用空格分开] APP_ABI := armeabi-v7a arm64-v8a
编译JNI
进入
BiuJniStatic/jni
输入
ndk-build
如果ndk配置没问题,根目录会出现libs目录,也就生成对应的so库。
build.gradle配置so库
# BiuJniStatic/build.gradle android { compileSdkVersion 31 // 略 sourceSets { main { jniLibs.srcDirs = ['libs'] } } }
编译运行,就可以测试Java调用JNI方法了。
参考文章
《》
《》
《》
© 版权声明