作为一个Andoird的Java程序员,会受到Java语言的局限,因为作为一面门向对象的语言不能像C/C++那样轻易调用与硬件有关的操作。因此JNI就搭建了这样一个桥梁,使Java和C/C++语言之间可以互相调用。作为一个Java工程师对C/C++的语言不是很熟悉,但只需熟悉他们之间调用的原理和方法,关于C/C++的编程就交给C语言工程师去吧。
在这篇文章中主要介绍NDK/JIN搭建和基本使用方法。
一、 环境的搭建
二、 基本的使用
步骤:
(1)新建Android工程
(2)在java中声明native方法
(3)在工程中新建jni文件夹(使用javah test 生成头文件,编写C代码)
(4)编写Android.mk文件(是一个配置文件告诉NDK如何编译C代码)
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := test-jni
LOCAL_SRC_FILES := test-jni.c
LOCAL_LDLIBS+=-llog
include $(BUILD_SHARED_LIBRARY)
(5)执行"ndk-build"生成动态库(在eclipse使用run也可以生成动态库)
(6)java代码"load"动态库,调用动态库
1、java调用C
java 调用native method(Class param)
在c中会调用的方法中参数(JNIEnv*,Jobject,Jint...)
2、c调用java
先用java调用c,c然后回调
JNIEXPORT void JNICALL Java_com_ndk2_test_ProvideBean_callHi(JNIEnv *env,
jobject obj) {
char*classname = "com/ndk2/test/ProvideBean";
jclass clazz;
clazz = (*env)->FindClass(env, classname);
if (clazz == 0) {
LOGI("can not find class");
} else {
LOGI("find the class");
}
jmethodID mid = (*env)->GetMethodID(env, clazz, "showHi", "()V");
if (mid == 0) {
LOGI("can not find method");
} else {
LOGI("find method");
}
(*env)->CallVoidMethod(env, obj, mid);
}
JNIEXPORT void JNICALL Java_com_ndk2_test_ProvideBean_calladd(JNIEnv *env,
jobject obj) {
char*classname = "com/ndk2/test/ProvideBean";
jclass clazz;
clazz = (*env)->FindClass(env, classname);
if (clazz == 0) {
LOGI("can not find class");
} else {
LOGI("find the class");
}
jmethodID mid = (*env)->GetMethodID(env, clazz, "showAdd", "(II)V");
if (mid == 0) {
LOGI("can not find method");
} else {
LOGI("find method");
}
(*env)->CallVoidMethod(env, obj, mid, 6, 7);
}
执行项目之前还要配置环境:
c/c++ Bulid Build command: bash D:\cygwin64\android-ndk-r8c\ndk-build
c/c++ General/Paths and Symbols/GNU C add D:\cygwin64\android-ndk-r8c\platforms\android-14\arch-arm\usr\include
如果还在报一些编译环境的错误,把工程转移到D:\cygwin64\android-ndk-r8c\samples基本就可以通过,这里我也不是很清楚是为什么,估计是路径映射的问题,哪位大神知道,可以分享下。