【网学网提醒】:网学会员编辑为广大网友搜集整理了:Android JNI实例(android 学习资料 集成测试 及实例程...绩等信息,祝愿广大网友取得需要的信息,参考学习。
Android的SDK中没有包括JNI的支持,而且对如何支持JNI也没有任何文档说明。不过既然整个Android平台是开源的,我们可以通过Google发布的源代码来找到一些线索(比如frameworks/base/media/jni/目录),依葫芦画瓢的实现上层JAVA程序通过JNI来调用NativeC
程序中的函数。
依照下面的步骤可以实现一个非常简单的JNI的实例
程序:
1.首先编写C模块,实现动态库。(关于如何在Android中编译C模块的更多细节,请参考《Android编译环境(1)
-编译NativeC的helloworld模块》。)
在development目录下添加新目录hellolib,并添加hellolib.c和Android.mk文件。hellolib.c的内容如下:
#include
#defineLOG_TAG"TestLib"
#undefLOG
#include
JNIEXPORTvoidJNICALLJava_com_test_TestHelloLib_printHello(JNIEnv*env,jobjectjobj)
{
LOGD("HelloLIB!\n");
}
注意这里的函数名需要按照JNI的规范(因此也可以用javah-jni工具来生成头文件,来保证函数名的正确性),Java_com_test_TestHelloLib_printHello的命名对应后面在java代码中,package名字是com.test,类名是TestHelloLib,native函数名是printHello。
另外,LOGD及#defineLOG_TAG"TestLib"等打印log的方式是采用了Android所提供的LOG机制,这样才能通过Android的logcat工具看到log。
用于编译C模块的Android.mk文件内容如下:
LOCAL_PATH:=$(callmy-dir)
include$(CLEAR_VARS)
LOCAL_SRC_FILES:=\
hellolib.c
LOCAL_C_INCLUDES:=\
$(JNI_H_INCLUDE)
LOCAL_SHARED_LIBRARIES:=\
libutils
LOCAL_PRELINK_MODULE:=false
LOCAL_MODULE:=libhello
include$(BUILD_SHARED_LIBRARY)
该文件中的一些变量分别对应的含义如下:
LOCAL_SRC_FILES-编译的源文件
LOCAL_C_INCLUDES-需要包含的头文件目录
LOCAL_SHARED_LIBRARIES-链接时需要的外部库
LOCAL_PRELINK_MODULE-是否需要prelink处理(参考prelink的详细介绍:《动态库优化——Prelink(预连接)技术》Android的Toolchain,prelink工具:Android,《
Toolchain与BionicLibc》)
LOCAL_MODULE-编译的目标对象
BUILD_SHARED_LIBRARY-指明要编译成动态库。
接下来回到Android顶层目录,并执行makelibhello来编译:
#cd$(YOUR_ANDROID)&;&;makelibhello
targetthumbC:libhello<=development/hellolib/hellolib.c
targetSharedLib:libhello(out/target/product/generic/obj/SHARED_LIBRARIES/libhello_intermediat
es/LINKED/libhello.so)
targetNon-prelinked:libhello(out/target/product/generic/symbols/system/lib/libhello.so)
targetStrip:libhello(out/target/product/generic/obj/lib/libhello.so)
Install:out/target/product/generic/system/lib/libhello.so
编译结果可得到位于out/target/product/generic/system/lib/目录的动态共享库libhello.so
2.编写Java模块,来通过JNI方式调用C接
口。具体Eclipse环境的搭建请参考AndroidSDK文档中的详细说明,及HelloAndroid程序的创建过程,这里仅给出我们需要修改的TestHelloLib.java文件:
packagecom.test;
importandroid.app.Activity;
importandroid.os.Bundle;
publicclassTestHelloLibextendsActivity{
/**Calledwhentheactivityisfirstcreated.*/
@Override
publicvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
printHello();
}
static{
System.loadLibrary("hello");
}
privatenativevoidprintHello();
}
注意上面代码中粗体字部分:privatenativevoidprintHello()用来声明一个native接口,static{System.loadLibrary("hello");}用来加载上面步骤中生成libhello.so(注意loadLibrary方法的参数不是”libhello.so”,而是去掉前缀和后缀之后的”hello”)onCreate()方法中则调用了printHello(),接口。
通过这一步骤可生成Android开发者所熟悉的apk文件:TestHelloLib.apk。
3.集成测试TestHelloLib.apk和libhello.so。先运行emulator并将TestHelloLib.apk
和libhello.so
上传至emulator中。注意要将libhello.so上传到emulator的
/system/lib目录,由于该目录是只读的,上传之前先要执行adbremount:
#adbremount
#adbpushout/target/product/generic/system/lib/libhello.so/system/lib
#adbinstallTestHelloLib.apk
接下来在模拟器菜单中可以看到已经安装的TestHelloLib程序,运行即可。
由于JNI接口printHello()并没有作界面上的改动,要验证其效果需要用
Android的logcat工具来查看。运行”adblogcat”可以找到下面的log片断:
I/ActivityManager(
48):Startingactivity:Intent
{action=android.intent.action.MAINcategories={android.intent.category.LAUNCHER}flags=0x10200000comp={com.test/com.test.TestHelloLib}}
I/ActivityManager(
48):Startproccom.testforactivity
com.test/.TestHelloLib:pid=174uid=10024gids={}
D/dalvikvm(
174):Tryingtoloadlib/system/lib/libhello.so0x43481c58
D/dalvikvm(
174):Addedsharedlib/system/lib/libhello.so
0x43481c58
D/dalvikvm(
174):NoJNI_OnLoadfoundin/system/lib/libhello.so0x43481c58
D/dalvikvm(
174):+++notscanning'/system/lib/libwebcore.so'for'printHello'(wrongCL)
D/dalvikvm(
174):+++notscanning'/system/lib/libmedia_jni.so'for'printHello'(wrongCL)
LIB!D/TestLib(174):HelloLIB!I/ActivityManager(48):Displayedactivity
com.test/.TestHelloLib:806ms
这里包含了调用printHello()接口的log信息,其中”D/TestLib(174):HelloLIB!”LIB!”就是D/TestLibprintHello()所打印的信息。至此成功完成AndroidJNI的实例验证。