JNI在Android开发中使用的比较广泛,因为Android应用层是用Java实现,底层是c/c++实现的,所以应用层调用底层库时需要使用JNI。如果你熟悉java和c/c++的话,那么学习JNI主要需要掌握java和c/c++数据类型的转换、JNI语法和函数编写规则。下面首先介绍java基本类型和引用类型跟JNI本地相关类型的对照,然后完成一个简单的demo。
1. 对照表
Java类型 本地类型 描述
boolean jboolean C/C++8位整型
byte jbyte C/C++带符号的8位整型
char jchar C/C++无符号的16位整型
short jshort C/C++带符号的16位整型
int jint C/C++带符号的32位整型
long jlong C/C++带符号的64位整型e
float jfloat C/C++32位浮点型
double jdouble C/C++64位浮点型
Object jobject 任何Java对象,或者没有对应java类型的对象
Class jclass Class对象
String jstring 字符串对象
Object[] jobjectArray 任何对象的数组
boolean[] jbooleanArray 布尔型数组
byte[] jbyteArray 比特型数组
char[] jcharArray 字符型数组
short[] jshortArray 短整型数组
int[] jintArray 整型数组
long[] jlongArray 长整型数组
float[] jfloatArray 浮点型数组
double[] jdoubleArray 双浮点型数组
2.打开eclipse,创建一个Android应用,名称:AndroidNDKTest
(1)打开自动生成的MainActivity.java,添加内容如下:
- public class MainActivity extends Activity {
- /** Called when the activity is first created. */
- @Override
- public void onCreate(Bundle savedInstanceState)
- {
- super.onCreate(savedInstanceState);
- TextView tv = new TextView(this);
- tv.setText( stringFromJNI() );
- setContentView(tv);
- printString("hello");
- }
- /**
- * 从c/c++文件中返回字符串
- * @return
- */
- public native String stringFromJNI();
- /**
- * 在c/c++文件中打印log信息
- * @param info
- */
- public native void printString(String info);
- static {
- System.loadLibrary("hello-jni");
- }
- }
上面代码中,我们声明了两个本地函数,在返回值前添加了native标志,表明是在c/c++定义实现的。
(2)在项目根目录下创建一个jni文件夹,创建一个hello-jni.c文件,内容如下:
- #include <string.h>
- #include <jni.h>
- #include <android/log.h>
- JNIEXPORT jstring JNICALL
- Java_com_example_ndktest_MainActivity_stringFromJNI( JNIEnv* env,
- jobject thiz )
- {
- return (*env)->NewStringUTF(env, "Hello from JNI !");
- }
- JNIEXPORT void JNICALL
- Java_com_example_ndktest_MainActivity_printString( JNIEnv* env,jobject thiz,jstring info)
- {
- const jchar* strDest;
- strDest = (*env)->GetStringUTFChars(env,info,NULL);
- if(strDest == NULL)
- {
- return NULL;
- }
- __android_log_print(ANDROID_LOG_INFO, "printString", strDest);
- (*env)->ReleaseStringUTFChars(env,info,strDest);
- }
JNIEXPORT和JNICALL都是JNI的关键字,是一些宏,这里不写也可以(JNIEXPORT jstring JNICALL可以改成jstring)。java的String对象对应于jni的jstring类型,但是在本地方法中不能直接使用,需要先转换成char*,这里使用GetStringUTFChars方法将传进来的jstring类型转换成为UTF-8格式的char*,就能够在本地方法中使用了。注意:使用完你所转换之后的对象之后,需要显示调用ReleaseStringUTFChars方法,让JVM释放转换的对象的空间,如果不显示的调用的话,该对象不会被垃圾回收器回收,因此可能会导致内存溢出。
下面是访问String的一些方法:
GetStringUTFChars 将jstring转换成为UTF-8格式的char*
GetStringChars 将jstring转换成为Unicode格式的char*
ReleaseStringUTFChars 释放指向UTF-8格式的char*的指针
ReleaseStringChars 释放指向Unicode格式的char*的指针
NewStringUTF 创建一个UTF-8格式的String对象
NewString 创建一个Unicode格式的String对象
GetStringUTFLength 获取UTF-8格式的char*的长度
GetStringLength 获取Unicode格式的char*的长度
(3) 添加一个Android.mk,相当于makefile,用来实现自动化编译的。内如如下:
- LOCAL_PATH := $(call my-dir)
- include $(CLEAR_VARS)
- LOCAL_LDLIBS := -llog
- LOCAL_MODULE := hello-jni
- LOCAL_SRC_FILES := hello-jni.c
- include $(BUILD_SHARED_LIBRARY)
这里添加了LOCAL_LDLIBS := -llog和hello-jni.c中添加了#include <android/log.h>头文件,是为了实现在c/c++代码中打印log信息的,以便在logcat中查看。
(4)配置ndk编译环境,这个在我的这篇文章里有详细说明。
(5)clean,然后运行程序,如果不出错,结果如下图:
模拟器
LogCat打印信息