好久不写jni,很多东西忘得差不多了。而且Android Studio下的jni开发和eclipse下的不太一样,记录一下一些小细节。
1、build.gradle的编写
Android Studio下的jni编译需要在build.gradle下指定编译方法和参数,目前支持cmake和ndkbuild编译,目前我还是用ndkbuild,例子如下:
defaultConfig { minSdkVersion 15 targetSdkVersion 24 versionCode 1 versionName "1.0" /*ndk { moduleName "base64" abiFilters "armeabi", "armeabi-v7a", "x86" }*/ externalNativeBuild { ndkBuild { abiFilters "armeabi", "armeabi-v7a", "x86" cFlags "-Wno-error=format-security" cppFlags "-std=c++11" } } } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } externalNativeBuild{ ndkBuild{ path "src/main/jni/Android.mk" } }注意上面两个externalNatibeBuild的位置和内容是不一样的
2、需要自己在src/mian/java下建立jni包,并在jni包下建立Android.mk文件
3、使用javah生成头文件:由于我是在ubuntu下开发的,所以cd到src/main/java目录下,假设需要生成头文件的类所在的包是com.exam,其类名为test,则生成头文件的命令为javah -jni com.exam.test
也可以加入-d参数指定头文件生成的位置:javah -d ./jni com.exam.test
javah会自动对java文件里修饰了native的方法生成头文件。其实如果只是写单一的jni不用头文件也行,不过生成头文件里的方法可以省略掉你自己敲方法名,避免出错。
注意,在cpp下写jni,要使用extern "C"包围代码
4、注意生成的头文件方法里的JNIEXPORT,JNICALL,JNIEnv* 和 jobject这四个
JNIEXPORT 和 JNICALL是jni.h中的宏,确保函数对外可见,可被c编译器调用编译。
JNIEnv接口指针,指向一个个函数表,函数表中的每一个入口指向一个JNI函数。本地方法经常通过这些函数来访问JVM中的数据结构
jobject指java中该native方法所在的类本身。
这2个宏和2个参数缺一不可,在任何一个jni方法中都必须存在。
5、jni中使用android的log方法
首先在Android.mk里加入LOCAL_LDLIBS := -llog
然后可以在头文件里定义宏:
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO ,LOG_TAG,__VA_ARGS__) #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR ,LOG_TAG,__VA_ARGS__) #define LOGV(...) __android_log_print(ANDROID_LOG_VERBOSE ,LOG_TAG,__VA_ARGS__)这样就可以在引用了头文件的源文件里随意调用LOGI、LOGE、LOGV之类的打印方法了。
注意在c++里,字符串的处理不像java那样简单,所以要活用string.h,sprintf之类的
6、在jni中,所有调用了java传入的参数的变量在最后一定要调用对应的release。诸如此类的还有c++的各种指针、数组等变量要在不需要的时候清空。