Android项目中JNI技术生成并调用.so动态库实现详解

摘要:
C/C++头文件将声明要调用的本机函数描述。然后我需要在Android项目的根目录中创建Application.mk文件:APP_project_PATH:=$APP_MODULES:=TestJNI编写了这两个mk文件后,我们可以使用ndk为我们生成相应的动态链接库。so文件

   生成 jni方式有两种:一种是通过SWIG从C++代码生成过度的java代码;另一种是通过javah的方式从java代码自动生成过度的C++代码。两种方式下的步骤流程正好相反。

第一种方式:由于需要配置SWIG环境,有点麻烦了,所以往往大家不采用这个途径,参照博文http://my.oschina.net/liusicong/blog/314162

第二种方式:javah的方式则通过shell指令就可以完成整个流程,该过程大概包括以下步骤:

  1. 编写 Java 代码。我们将从编写 Java 类开始,这些类执行三个任务:声明将要调用的本机方法;装入包含本机代码的共享库;然后调用该本机方法。
  2. 编译 Java 代码。在使用 Java 类之前,必须成功地将它们编译成字节码。

  3. 创建 C/C++ 头文件。C/C++     头文件将声明想要调用的本机函数说明。然后,这个头文件与 C/C++ 函数实现(请参阅步骤 4)一起来创建共享库(请参阅步骤 5)。

  4. 编写 C/C++ 代码。这一步实现 C 或     C++ 源代码文件中的函数。C/C++ 源文件必须包含步骤 3 中创建的头文件。

  5. 创建共享库文件。从步骤 4 中创建的 C 源代码文件来创建共享库文件。

  6. 运行 Java 程序。运行该代码,并查看它是否有用。我们还将讨论一些用于解决常见错误的技巧。

第二种方法的具体实现方式和例子如下:

1. 在Eclipse中创建项目:TestJNI

2. 新创建一个class:TestJNI.java

package com.wwj.jni;
 public class TestJNI {    
    public native boolean Init();    
    public native int Add(int x, int y);    
    public native void Destory();
}

以上代码声明三个本地方法。

3. 编译JNI

找到Android项目中bin目录下,会有classes文件夹,Eclipse自动为我们生成的字节码文件就在这个目录下。

我们在该路径下,使用javah命令,生成我们想要得到的.h头文件,如下图所示:

执行javah -jni com.wwj.jni.TestJNI命令之后,会在classes目录下生成头文件:com_wwj_jni_TestJNI.h

将它复制到jni文件夹下,打开如下:

复制代码
/* DO NOT EDIT THIS FILE - it is machine generated */
 #include <jni.h>
/* Header for class com_wwj_jni_TestJNI */
 #ifndef _Included_com_wwj_jni_TestJNI
 #define _Included_com_wwj_jni_TestJNI
 #ifdef __cplusplusextern "C" {
#endif
/*
 * Class:     com_wwj_jni_TestJNI
 * Method:    Init
 * Signature: ()Z
 */
 JNIEXPORT jboolean JNICALL Java_com_wwj_jni_TestJNI_Init
  (JNIEnv *, jobject);
/*
 * Class:     com_wwj_jni_TestJNI
 * Method:    Add
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_wwj_jni_TestJNI_Add
  (JNIEnv *, jobject, jint, jint);
/*
 * Class:     com_wwj_jni_TestJNI
 * Method:    Destory
 * Signature: ()V 
 */
 JNIEXPORT void JNICALL Java_com_wwj_jni_TestJNI_Destory
  (JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
复制代码

以上代码就是通过javah命令生成jni层代码。

4. 使用C/C++实现JNI

在jni文件夹下,创建com_wwj_jni_TestJNI.h对应的cpp文件:com_wwj_jni_TestJNI.cpp

我们再添加两个文件Add.h,Add.cpp,具体实现放在这两个文件中来完成。

Add.h

复制代码
#ifndef _TEST_JNI_ADD_H_
 #define _TEST_JNI_ADD_H_
    class CAdd {public:
         CAdd();    
        ~CAdd();    
        int Add(int x, int y);
    };
#endif
复制代码

Add.cpp

复制代码
#include "Add.h"
 CAdd::CAdd() {
 }
 CAdd::~CAdd() {
 }
 int CAdd::Add(int x, int y) {    
 return x + y;
}
复制代码

com_wwj_jni_TestJNI.cpp的实现:

复制代码
#include <stdio.h>
#include <stdlib.h>
#include "com_wwj_jni_TestJNI.h"
#include "Add.h"
Add *pCAdd = NULL;
JNIEXPORT jboolean JNICALL Java_com_wwj_jni_TestJNI_Init(JNIEnv *env,jobject obj) {   
   if (pCAdd == NULL) {
        pCAdd = new CAdd;
    }    
   return pCAdd != NULL;
}
JNIEXPORT jint JNICALL Java_com_wwj_jni_TestJNI_Add(JNIEnv *env, jobject obj,
        jint x, jint y) {    
        int res = -1;    
       if (pCAdd != NULL) {
          res = pCAdd->Add(x, y);
        }    
       return res;
}
JNIEXPORT void JNICALL Java_com_wwj_jni_TestJNI_Destory(JNIEnv *env, jobject obj)
{    if (pCAdd != NULL)
    {
        pCAdd = NULL;
    }
}
复制代码

5. 创建mk文件,并使用ndk-build命令生成.so动态链接库文件

在jni目录下创建Android.mk文件如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := TestJNI
LOCAL_SRC_FILES := com_wwj_jni_TestJNI.cpp
LOCAL_SRC_FILES += Add.cpp
include $(BUILD_SHARED_LIBRARY)

其中:

LOCAL_PATH是C/C++代码所在目录,也就是我们的jni目录。

LOCAL_MODULE是要编译的库的名称。编译器会自动在前面加上lib,在后面加上.so。

LOCAL_SRC_FILES是要编译的C/C++文件。

然后我还需要在Android项目根目录下创建Application.mk文件:

APP_PROJECT_PATH := $(call my-dir)
APP_MODULES := TestJNI

写完了这两个mk文件,我们就可以用ndk来为我们生成相应的动态链接库了。前提你需要下载NDK,并把NDK路径配置到path环境变量中去,笔者配置的路径是:D:Cocos2dxandroid-ndk-r9d,具体视个人情况而定。

进入Application.mk文件所在目录,在命令行中使用ndk-build生成.so文件

编译成功后会在工程目录的libs/armeabi目录下生成一个libTestJNI.so文件。

项目结构会变成如下:

6. 在Java中调用JNI

复制代码
package com.wwj.jni;
import android.os.Bundle;
import android.widget.TextView;
import android.app.Activity;
public class TestJNIActivity extends Activity {    
private TextView textView;    
static {        // 加载动态库
        System.loadLibrary("TestJNI");
    }
    @Override    
protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = (TextView) findViewById(R.id.textview);
        TestJNI testJNI = new TestJNI();        // 调用native方法
        boolean init = testJNI.Init();        
       if (init == true) {            // 调用Add函数
            int sum = testJNI.Add(100, 150);
            textView.setText("你真是个" + sum);
        } else {
            textView.setText("你比二百五还要二百五");
        }
        testJNI.Destory();
    }
}
复制代码

运行项目,效果图如下:

按照以上步骤完全可以生成一个属于自己的.so文件,且可正确调用和执行,这里补充一下两点:

注意一:NDK的环境配置:

1,下载地址:

Android NDK r10e:

    32位:http://dl.google.com/android/ndk/android-ndk-r10e-windows-x86.exe

    64位:http://dl.google.com/android/ndk/android-ndk-r10e-windows-x86_64.exe

Android NDK r9d:

    32位:https://dl.google.com/android/ndk/android-ndk-r9d-windows-x86.zip

    64位: https://dl.google.com/android/ndk/android-ndk-r9d-windows-x86_64.zip

自己用的是ndk-r9d版本,以上地址需要FQ,需要的对应版本留下邮箱就好。

2,配置PATH环境变量,cmd命令行输入ndk-build,出现一下内容提示表示配置正确。

Android NDK: Could not find application project directory !
Android NDK: Please define the NDK_PROJECT_PATH variable to point to it.
D:javaandroid-ndk-r9duild/core/build-local.mk:148: *** Android NDK: Aborting
    .  Stop.

注意二:参考以上文章在最后输入ndk-build时会提示以下错误:

此时修改一下Application.mk文件里的:

APP_PROJECT_PATH := $(call my-dir)/ 成 APP_PROJECT_PATH := $(call my-dir)/..

文章转载于:http://www.cnblogs.com/sevenyuan/p/4202759.html

免责声明:文章转载自《Android项目中JNI技术生成并调用.so动态库实现详解》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇日记账导入APIDjango用Apache和mod_wsgi部署下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Android JNI和NDK学习(08)JNI实例一 传递基本类型数据

Android JNI和NDK学习(08)--JNI实例一 传递基本类型数据 本文介绍在Java和JNI之间相互传递基本数据类型的方法。 由于前面已经详细介绍搭建和建立NDK工程的完整流程(参考“静态实现流程”或“动态实现流程”),这里就不再介绍流程;而是将重点放在说明如何实现Java和JNI之间相互传递基本数据。 1 建立eclipse工程 建立工程Nd...

Android.mk 文件语法详解

0. Android.mk简介: Android.mk文件用来告知NDK Build 系统关于Source的信息。 Android.mk将是GNU Makefile的一部分,且将被Build System解析一次或多次。 所以,请尽量少的在Android.mk中声明变量,也不要假定任何东西不会在解析过程中定义。 Android.mk文件语法允许我们将Sou...

【Delphi】RAD 10.4 开发Android时如何在迁移SDK和NDK路径后解决提示 ld: cannot find -lxxx的问题

今天使用lite版本安装了RAD Delphi10.4,发现自动把Android的SDK和NDK目录安装到C盘下,由于C盘容量紧张,所以想把目录迁移到其他路径! 在之前的版本是直接移动目录,在到delphi的tools-options菜单页面中修改SDK路径即可。 但是这次修改后发现在link时仍然找不到路径,查看了delphi的编译命令行信息,发现其中...

Android之 MTP框架和流程分析

概要 本文的目的是介绍Android系统中MTP的一些相关知识。主要的内容包括:第1部分 MTP简介对Mtp协议进行简单的介绍。第2部分 MTP框架介绍Android系统下MTP的框架。第3部分 MTP启动流程详细分析MTP服务的启动流程,包括Java层, JNI层, kernel相关知识的介绍。第4部分 MTP协议之I->R流程以"PC中打开一个M...

jni中关于dll的装载问题[转]

   通常我们在写大型项目时,也就是集成的项目时,单独用JAVA语言去完成所有的事情往往效果不佳,也不能很好的完成,例如:我们要去调硬件,获取电脑的运行状况等等(如果用JAVA语言实现时,往往耗时),基于此,我们就要寻求一种很好的解决方案,那就是利用别的语言的长处了,如:C++(它在对底层的调用和硬件方面确实够强悍)。如果我们能用C++实现对硬件的所有操作...

11.Unity3D与android交互---构建android插件

为android构建一个插件 要创建一个android插件,首先要有 Android NDK 并熟悉使用ndk构建共享库的方法。 如果用C++来实现库,必须声明成用C语言的链接方式,以避免Name Mangling问题。 extern "C" { float FooPluginFunction (); } 通过C#脚本使用插件 构建了共享库后,必...