前言
在上一篇《》的动态注册并调用Java方法,上次也说了可以把公共部分封装一下。
说干就干,今天就把上次代码封装一下。
正文
我这不介绍完整的动态注册,有需要可以看《》。这里只是个人流水账。
Hello.java
- package com.biumall.dynamic.one;
- public class Hello {
- //1. load Hello.so
- static {
- System.loadLibrary("Hello");
- }
- // 2. define native computer()
- public static native int computer(String type, int a, int b);
- //add
- public int add(int x, int y) {
- return x + y;
- }
- //sub
- public int sub(int x, int y) {
- return x - y;
- }
- }
JNI
把之前Hello.c的代码拆分为Log.h,JNICommon.h和Hello.c,也就是吧公共部分单独抽出,这样移植就方便多了。
Log.h
- #include <android/log.h>
-
- //定义TAG日志打印
- #define LOG_TAG "from_dynamic_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__)
网上有封装Log等级控制,方便控制日志打印,目前没空,后续补上。
JNICommon.h
就获取数组大小和本地方法注册。
- #include <jni.h>
-
- //获取数组大小
- #define ARRAY_LENGTH(x) ((int)(sizeof(x) / sizeof((x)[0])))
-
- //注册本地方法
- int registerNativeMethods(JNIEnv *env, const char *className, JNINativeMethod *methods,int methods_size) {
- jclass clazz = (*env)->FindClass(env, className);
- if (!clazz) {
- return JNI_ERR;
- }
- int result = (*env)->RegisterNatives(env, clazz, methods, methods_size);
- if (result < 0) {
- return JNI_ERR;
- }
- return JNI_OK;
- }
Hello.c
特意把每步单独隔开,方便初学者(我就是)查看。
- #include "JNICommon.h"
- #include "Log.h"
- #include<string.h>
-
- /**
- * ---------------------------第一步 start ---------------------------
- * 定义Hello.java类路径[包名+类名,只不过.换成了/]
- * 包名+类名:com.biumall.dynamic.one.Hello
- */
- #define DYNAMIC_CLASS "com/biumall/dynamic/one/Hello"
- /**
- * ---------------------------第一步 end ---------------------------
- */
-
- /**
- * ---------------------------第二步 start ---------------------------
- * c_add()用于调用Java方法
- * @param env env
- * @param jclazz jclazz
- * @param x x
- * @param y y
- * @return
- */
- int c_add(JNIEnv *env, jclass jclazz, int x, int y) {
- jmethodID methodID = (*env)->GetMethodID(env, jclazz, "add", "(II)I");
- jobject object = (*env)->AllocObject(env, jclazz);
- return (*env)->CallIntMethod(env, object, methodID, x, y);
- }
-
- /**
- * c_sub()用于调用Java方法
- * @param env env
- * @param jclazz jclazz
- * @param x x
- * @param y y
- * @return
- */
- int c_sub(JNIEnv *env, jclass jclazz, int x, int y) {
- jmethodID methodID = (*env)->GetMethodID(env, jclazz, "sub", "(II)I");
- jobject object = (*env)->AllocObject(env, jclazz);
- return (*env)->CallIntMethod(env, object, methodID, x, y);
- }
-
- /**
- * 本地方法computer实现
- */
- JNIEXPORT jint JNICALL native_computer(JNIEnv *env, jclass jclazz, jstring type, jint a, jint b) {
- const char *charType = (*env)->GetStringUTFChars(env, type, 0);
- int count;
- if (!strcmp(charType, "-")) {
- count = c_sub(env, jclazz, a, b);
- } else if (!strcmp(charType, "+")) {
- count = c_add(env, jclazz, a, b);
- }
- (*env)->ReleaseStringUTFChars(env, type, charType);
- return count;
- }
- /**
- * ---------------------------第二步 end ---------------------------
- */
-
- /**
- * ---------------------------第三步 start ---------------------------
- * 定义Java和JNI函数的绑定表
- * 方法数组,分别为:(native方法 , 方法签名 , 函数指针[JNI中需要实现的方法名])
- * 通过javac和javap 获取 方法签名
- */
- JNINativeMethod method_table[] = {
- {"computer", "(Ljava/lang/String;II)I", (void *) native_computer},
- };
- /**
- * ---------------------------第三步 end ---------------------------
- */
-
- /**
- * ---------------------------第四步 start ---------------------------
- * JNI_OnLoad
- */
- JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved ){
- JNIEnv *env = NULL;
- int result = (*vm)->GetEnv(vm, (void **) &env, JNI_VERSION_1_6);
- LOGD("JNI_OnLoad 1 result : %d", result);
- if(result != JNI_OK){
- return JNI_ERR;
- }
- result = registerNativeMethods(env, DYNAMIC_CLASS, method_table, ARRAY_LENGTH(method_table));
- LOGD("JNI_OnLoad 2 result : %d", result);
- if(result != JNI_OK){
- return JNI_ERR;
- }
- return JNI_VERSION_1_6;
- }
- /**
- * ---------------------------第四步 end ---------------------------
- */
参考文章
《》
© 版权声明