Android JNI实例

转载 2010年06月02日 16:41:00

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 <jni.h>

 

#define LOG_TAG "TestLib"

#undef LOG

#include <utils/Log.h>

 

 

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

{

    LOGD("Hello LIB!/n");

}

注意这里的函数名需要按照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 {

    /** Called when the activity is first created. */

    @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的实例验证。

文章来源:www.top-e.org

Android视频《JNI》

-
  • 1970年01月01日 08:00

Android NDK JNI 经典实例

  • 2011年04月14日 15:44
  • 85KB
  • 下载

eclipse下android JNI开发小实例

JNI是JavaNativeInterface的缩写,通过JNI可以方便我们在Android平台上进行C/C++编程。要用JNI首先必须安装Android的NDK,配置好NDK环境之后就可以在Ecli...
  • sky1203850702
  • sky1203850702
  • 2014-12-13 10:51:20
  • 4268

Android中JNI创建实例

参考文档: http://blog.sina.com.cn/s/blog_a11f64590101924l.html http://www.cnblogs.com/hoys/archive/2010/...
  • wikiday
  • wikiday
  • 2015-01-04 22:18:46
  • 9131

Android studio简单使用JNI实例

Android studio简单使用JNI实例 发现网上很多JNI的使用教程,也很详细,不过有的地方有些缺漏,导致很多小问题难以解决的,今天就来总结一下。   准备工作:下载NDK。...
  • wangliu1102
  • wangliu1102
  • 2017-12-28 10:58:55
  • 637

Android Jni使用案例

Android上层和C/C++层通信可以通过JNI实现,具体做法有两种: 一:采用默认的本地函数注册流程   1). 编写带有native方法的Java类;     Gpio.java ...
  • visionliao
  • visionliao
  • 2016-05-26 15:12:21
  • 1592

Android JNI 实例

  • 2017年08月30日 16:32
  • 111KB
  • 下载

Android-使用Android studio 进行NDK-JNI开发实例(一)

前言 使用Android studio 进行NDK-JNI开发,在界面上显示Hello NDK-JNI 步骤: 1. 创建工程 2. 配置NDK 路径 local.properties 文件中加入 n...
  • u014657752
  • u014657752
  • 2015-08-30 19:03:28
  • 5033

NDK-JNI实战教程(三) 从比Hello World稍复杂点儿的NDK例子说说模板

PS一句:最终还是选择CSDN来整理发表这几年的知识点,该文章平行迁移到CSDN。因为CSDN也支持MarkDown语法了,牛逼啊!【工匠若水 http://blog.csdn.net/yanbobe...
  • yanbober
  • yanbober
  • 2015-04-27 15:08:46
  • 13710

【Android应用开发】-(14)JNI----经典实例分析

接下来几篇文章,学习JNI开发,主要是学习《JNI详解》,将里面的实例通过Android平台实现,首先由一个经典实例引出 – HelloWorld!!HelloWorld是经典的,一想起学编程,就想起...
  • tangcheng_ok
  • tangcheng_ok
  • 2012-06-27 16:59:54
  • 3936
收藏助手
不良信息举报
您举报文章:Android JNI实例
举报原因:
原因补充:

(最多只允许输入30个字)