Android NDK 学习之调用Java函数

时间:2022-09-09 05:05:27

本博客主要是在Ubuntu 下开发,且默认你已经安装了Eclipse,Android SDK, Android NDK, CDT插件。

在Eclipse中添加配置NDK,路径如下Eclipse->Window->Preferences->Android->NDK ,选择NDK的路径,然后Apply即可。

新建一个名为AndroidJNI_MethodCall的Android工程,新建一个jni的文件夹,其目录下文件树列表如下:

├── jni
│   ├── Android.mk
│   ├── Application.mk
│   └── methodcall
│      ├── Android.mk
│      ├── logger.h
│      └── methodcall_jni.c

jni/Application.mk文件内容如下:

APP_ABI := all

jni/Android.mk,主要用来指定顺序执行所有子文件夹下面的makefile文件,内容如下:

include $(call all-subdir-makefiles)

jni/methodcall/Android.mk,主要用来指定需要编译出的动态库的名称,以及需要编译的源文件,内容如下:

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE     := methodcall_jni
LOCAL_SRC_FILES := methodcall_jni.c LOCAL_LDLIBS := -llog include $(BUILD_SHARED_LIBRARY)

jni/methodcall/logger.h 主要用来在JNI层打印日志,内容如下:

#include <jni.h>
#include <android/log.h> /**
* 定义log标签
*/
#define TAG "jni_logger" /**
* 定义info信息
*/
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__) /**
* 定义debug信息
*/
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, TAG, __VA_ARGS__) /**
* 定义error信息
*/
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, TAG, __VA_ARGS__)

jni/methodcall/methodcall_jni.c,主要用来注册绑定java函数和native函数,以及java函数在c中相应函数的具体实现, 内容如下:

#include "logger.h"

#ifndef NULL
#define NULL ((void *) 0)
#endif /**
* 获取数组的大小
*/
#define NELEM(x) ((int) (sizeof(x) / sizeof((x)[0]))) /**
* 指定要注册的类,对应的完整的java类名
*/
#define JNIREG_CLASS "com/clarck/jni/MethodCall" JNIEXPORT void JNICALL native_Method(JNIEnv *env, jobject obj) {
/*jclass cls = (*env)->GetObjectClass(env, obj);
//在被给的类中方法执行一个查询。查询是依靠名字和方法的类型描述符。
jmethodID mid = (*env)->GetMethodID(env, cls, "callback", "()V");
if (NULL == mid) {
return ;
}
LOGI("In C \n");
//然后本地代码调用"CallVoidMethod"。"CallVoidMethod"调用一个实例方法,它返回一个 void类型。你传递一个对象(object),方法 ID 和实际参数
(*env)->CallVoidMethod(env, obj, mid);*/ jclass cls = (*env)->GetObjectClass(env, obj);
jmethodID mid = (*env)->GetStaticMethodID(env, cls, "_callback", "()V");
if (NULL == mid) {
return ;
} LOGI("In C \n");
(*env)->CallStaticVoidMethod(env, cls, mid);
} /**
* Java和JNI函数绑定
*/
static JNINativeMethod method_table[] = {
{ "nativeMethod", "()V", (void*) native_Method },
}; /**
* 注册native方法到java中
*/
static int registerNativeMethods(JNIEnv* env, const char* className,
JNINativeMethod* gMethods, int numMethods) {
jclass clazz;
clazz = (*env)->FindClass(env, className);
if (clazz == NULL) {
return JNI_FALSE;
} if ((*env)->RegisterNatives(env, clazz, gMethods, numMethods) < ) {
return JNI_FALSE;
} return JNI_TRUE;
} /**
* 调用注册方法
*/
int register_ndk_load(JNIEnv* env) {
return registerNativeMethods(env, JNIREG_CLASS, method_table,
NELEM(method_table));
} JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) {
JNIEnv* env = NULL;
jint result = -; if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK) {
return result;
} register_ndk_load(env); //返回jni的版本
return JNI_VERSION_1_4;
}

接着在Project中右键Android Tools->Add Native Support,最后java层调用如下:

package com.clarck.jni;

import android.util.Log;

public class MethodCall {
public native void nativeMethod(); public void callback() {
Log.d("Test", "In Android");
} public static void _callback() {
Log.d("Test", "In Android static");
} static {
System.loadLibrary("methodcall_jni");
}
}
package com.clarck.jni;

import android.app.Activity;
import android.os.Bundle; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); MethodCall methodCall = new MethodCall();
methodCall.nativeMethod();
}
}

执行Ctrl+B执行编译,运行工程即可。