博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android JNI 应用实例
阅读量:4051 次
发布时间:2019-05-25

本文共 4066 字,大约阅读时间需要 13 分钟。

Android的SDK中没有包括JNI的支持,而且对如何支持JNI也没有任何文档说明。不过既然整个Android平台是开源的,我们可以通过Google发布的源代码来找到一些线索(比如frameworks/base/media/jni/目录),依葫芦画瓢的实现上层JAVA程序通过JNI来调用Native C程序中的函数。

  依照下面的步骤可以实现一个非常简单的JNI的实例程序:

  1. 首先编写C模块,实现动态库。(关于如何在Android中编译C模块的更多细节,请参考《Android编译环境(1) - 编译Native C的helloworld模块》。)

  在development目录下添加新目录hellolib,并添加hellolib.c和Android.mk文件。hellolib.c的内容如下:

  #include

  #define LOG_TAG "TestLib"

  #undef LOG

  #include

  JNIEXPORT void JNICALL Java_com_test_TestHelloLib_printHello(JNIEnv * env, jobject jobj)

  {

  LOGD("Hello LIB! ");

  }

  注意这里的函数名需要按照JNI的规范(因此也可以用javah -jni工具来生成头文件,来保证函数名的正确性),Java_com_test_TestHelloLib_printHello的命名对应后面在java代码中,package名字是com.test,类名是TestHelloLib,native函数名是printHello。

  另外,LOGD及#define LOG_TAG "TestLib"等打印log的方式是采用了Android所提供的LOG机制,这样才能通过Android的logcat工具看到log。

  用于编译C模块的Android.mk文件内容如下:

  LOCAL_PATH:= $(call my-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与Bionic Libc》)

 

  LOCAL_MODULE - 编译的目标对象

  BUILD_SHARED_LIBRARY - 指明要编译成动态库。

  接下来回到Android顶层目录,并执行make libhello来编译:

  # cd $(YOUR_ANDROID) && make libhello

  target thumb C: libhello <= development/hellolib/hellolib.c

  target SharedLib: libhello (out/target/product/generic/obj/SHARED_LIBRARIES/libhello_intermediates/LINKED/libhello.so)

  target Non-prelinked: libhello (out/target/product/generic/symbols/system/lib/libhello.so)

  target Strip: 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环境的搭建请参考Android SDK文档中的详细说明,及Hello Android程序的创建过程,这里仅给出我们需要修改的TestHelloLib.java文件:

  package com.test;

  import android.app.Activity;

  import android.os.Bundle;

  public class TestHelloLib extends Activity {

  

  @Override

  public void onCreate(Bundle savedInstanceState) {

  super.onCreate(savedInstanceState);

  setContentView(R.layout.main);

  printHello();

  }

  static {

  System.loadLibrary("hello");

  }

  private native void printHello();

  }

  注意上面代码中粗体字部分:private native void printHello()用来声明一个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目录,由于该目录是只读的,上传之前先要执行adb

remount:

 

  # adb remount

  # adb push out/target/product/generic/system/lib/libhello.so /system/lib

  # adb install TestHelloLib.apk

  接下来在模拟器菜单中可以看到已经安装的TestHelloLib程序,运行即可。

  由于JNI接口printHello()并没有作界面上的改动,要验证其效果需要用Android的logcat工具来查看。运行”adb logcat”可以找到下面的log片断:

  I/ActivityManager( 48): Starting activity: Intent { action=android.intent.action.MAIN categories={android.intent.category.LAUNCHER} flags=0x10200000 comp={com.test/com.test.TestHelloLib} }

  I/ActivityManager( 48): Start proc com.test for activity com.test/.TestHelloLib: pid=174 uid=10024 gids={}

  D/dalvikvm( 174): Trying to load lib /system/lib/libhello.so 0x43481c58

  D/dalvikvm( 174): Added shared lib /system/lib/libhello.so 0x43481c58

  D/dalvikvm( 174): No JNI_OnLoad found in /system/lib/libhello.so 0x43481c58

  D/dalvikvm( 174): +++ not scanning '/system/lib/libwebcore.so' for 'printHello' (wrong CL)

  D/dalvikvm( 174): +++ not scanning '/system/lib/libmedia_jni.so' for 'printHello' (wrong CL)

  D/TestLib ( 174): Hello LIB!

  I/ActivityManager( 48): Displayed activity com.test/.TestHelloLib: 806 ms

  这里包含了调用printHello()接口的log信息,其中”D/TestLib ( 174): Hello LIB!”就是printHello()所打印的信息。至此成功完成Android JNI的实例验证。

转载地址:http://pkpci.baihongyu.com/

你可能感兴趣的文章
socket,accept函数解析
查看>>
今日互联网关注(写在清明节后):每天都有值得关注的大变化
查看>>
”舍得“大法:把自己的优点当缺点倒出去
查看>>
[今日关注]鼓吹“互联网泡沫,到底为了什么”
查看>>
[互联网学习]如何提高网站的GooglePR值
查看>>
[关注大学生]求职不可不知——怎样的大学生不受欢迎
查看>>
[关注大学生]读“贫困大学生的自白”
查看>>
[互联网关注]李开复教大学生回答如何学好编程
查看>>
[关注大学生]李开复给中国计算机系大学生的7点建议
查看>>
[茶余饭后]10大毕业生必听得歌曲
查看>>
VC++ MFC SQL ADO数据库访问技术使用的基本步骤及方法
查看>>
VUE-Vue.js之$refs,父组件访问、修改子组件中 的数据
查看>>
Vue-子组件改变父级组件的信息
查看>>
Python自动化之pytest常用插件
查看>>
Python自动化之pytest框架使用详解
查看>>
【正则表达式】以个人的理解帮助大家认识正则表达式
查看>>
性能调优之iostat命令详解
查看>>
性能调优之iftop命令详解
查看>>
非关系型数据库(nosql)介绍
查看>>
移动端自动化测试-Windows-Android-Appium环境搭建
查看>>