Android开发之NDK

作者: Reathin | 来源:发表于2017-04-10 22:24 被阅读336次

    NDK

    NDK全称:Native Development Kit。

    关于NDK,360百科是这么说的:
    1.NDK是一系列工具的集合。

    • NDK提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库,并能自动将so和java应用一起打包成apk。这些工具对开发者的帮助是巨大的。

    • NDK集成了交叉编译器,并提供了相应的mk文件隔离平台、CPU、API等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编译特性要求”等),就可以创建出so。

    • NDK可以自动地将so和Java应用一起打包,极大地减轻了开发人员的打包工作。

    2.NDK提供了一份稳定、功能有限的API头文件声明。

    Google明确声明该API是稳定的,在后续所有版本中都稳定支持当前发布的API。从该版本的NDK中看出,这些API支持的功能非常有限,包含有:C标准库(libc)、标准数学库(libm)、压缩库(libz)、Log库(liblog)。

    NDK产生的背景

    Android平台从诞生起,就已经支持C、C++开发。众所周知,Android的SDK基于Java实现,这意味着基于Android SDK进行开发的第三方应用都必须使用Java语言。但这并不等同于“第三方应用只能使用Java”。在Android SDK首次发布时,Google就宣称其虚拟机Dalvik支持JNI编程方式,也就是第三方应用完全可以通过JNI调用自己的C动态库,即在Android平台上,“Java+C”的编程方式是一直都可以实现的。
      不过,Google也表示,使用原生SDK编程相比Dalvik虚拟机也有一些劣势,Android SDK文档里,找不到任何JNI方面的帮助。即使第三方应用开发者使用JNI完成了自己的C动态链接库(so)开发,但是so如何和应用程序一起打包成apk并发布?这里面也存在技术障碍。比如程序更加复杂,兼容性难以保障,无法访问Framework API,Debug难度更大等。开发者需要自行斟酌使用。
      于是NDK就应运而生了,2011发布NDK。NDK全称是Native Development Kit。NDK的发布,使“Java+C”的开发方式终于转正,成为官方支持的开发方式。NDK将是Android平台支持C开发的开端。

    NDK作用

    • 代码的保护。由于apk的java层代码很容易被反编译,而C/C++库反汇难度较大。

    • 可以方便地使用现存的开源库。大部分现存的开源库都是用C/C++代码编写的。

    • 提高程序的执行效率。将要求高性能的应用逻辑使用C开发,从而提高应用程序的执行效率。(在前面性能优化中有提及)

    • 便于移植。用C/C++写得库可以方便在其他的嵌入式平台上再次使用。

    具体可参考NDK 入门指南

    JNI

    JNI是Java Native Interface的缩写,它提供了若干的API实现了Java和其他语言的通信(主要是C&C++)。从Java1.1开始,JNI标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互。JNI一开始是为了本地已编译语言,尤其是C和C++而设计的,但是它并不妨碍你使用其他编程语言,只要调用约定受支持就可以了。使用java与本地已编译的代码交互,通常会丧失平台可移植性。但是,有些情况下这样做是可以接受的,甚至是必须的。例如,使用一些旧的库,与硬件、操作系统进行交互,或者为了提高程序的性能。JNI标准至少要保证本地代码能工作在任何Java 虚拟机环境。

    通俗点的意思就是用JAVA调用C或者C++。在实际开发过程中很可能会使用到C或者C++开发的DLL(windows平台),或者so(Linux平台),这个时候就需要用JAVA来调用DLL或者so文件。

    开发NDK时,需要用到JNI。

    接口分析

    JNIEXPORT void JNICALL Java_com_test01_Test_firstTest (JNIEnv * env, jobject obj);

    • JNIEXPORT :在Jni编程中所有本地语言实现Jni接口的方法前面都有一个"JNIEXPORT",这个可以看做是Jni的一个标志,至今为止没发现它有什么特殊的用处。

    • void :这个学过编程的人都知道,当然是方法的返回值了。

    • JNICALL :这个可以理解为Jni 和Call两个部分,和起来的意思就是 Jni调用XXX(后面的XXX就是JAVA的方法名)。

    • Java_com_test01_Test_firstTest:这个就是被上一步中被调用的部分,也就是Java中的native 方法名,这里起名字的方式比较特别,是:包名+类名+方法名。

    • JNIEnv * env:这个env可以看做是Jni接口本身的一个对象,jni.h头文件中存在着大量被封装好的函数,这些函数也是Jni编程中经常被使用到的,要想调用这些函数就需要使用JNIEnv这个对象。例如:env->GetObjectClass()。

    Jni中的数据类型

    每一个Java的数据类型在Jni中都一个和它相对应的数据类型,这样才能保证Java调用C或者C++的过程中数据的正确性。jni.h头文件中定义的类型:

    ![2.png](http:https://img.haomeiwen.com/i4623465/45b6afc193305be2.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

    NDK环境搭建

    下载NDK,下载具体文件解压即可,也可使用studio的sdk manager下载安装

    Android-Studio配置

    配置NDK路径

    • 右击Module->Open Module Setting->SDK Location->Android NDK Location
    • 或者直接修改local.properties文件
    QQ图片20170410212255.png

    创建Library项目,定义模板类,此类主要为了生成so文件用,so文件生成后可删除

    // 包名和类名要和.cpp或者.c文件中一致
    package com.ndkdemo;
    
    public class MathKit
    {
       // 定义native本地方法,和普通方法相同,加上native关键字
        public static native int square(int num);
    }
    

    创建jni目录,默认为src/main/jni

    javah命令生成.h文件

    在Android Studio找到View->Tool Windows->Terminal 打开命令行:
    执行如下命令:

    javah -d NDKDemo/src/main/jni/ -classpath D:/AndroidStudioProjects/MyWork/NDKDemo/build/intermediates/classes/debug  -jni com.ndkdemo.MathKit
    

    或者

    cd D:/AndroidStudioProjects/MyWork/NDKDemo/build/intermediates/classes/debug
    javah com.ndkdemo.MathKit
    

    -d . 表示将在当前目录下生成一个当前命令行文件夹,产生的头文件就在这里面了;
    -classpath < PATH> 指明class文件所在的位置(目录)
    -jni com.ndkdemo.MathKit 指定类名

    javah命令主要用于在JNI开发的时,把java代码声明的JNI方法转化成C\C++ 头文件,以便进行JNI的C\C++ 端程序的开发。
    但是需要注意的是javah命令对Android编译生成的类文件并不能正常工作。如果对于Android的JNI要想生成C\C++ 头文件的话,可能只有先写个纯的java代码来进行JNI定义,接着用JDK编译,然后再用javah命令生成JNI的C\C++ 头文件。当然你也可以不用javah命令,直接手写JNI的C\C++ 头文件。

    创建cpp文件,文件名最好和.h文件同名,便于管理编辑.cpp文件

    #include <com_ndkdemo_MathKit.h>
    
    JNIEXPORT jintJNICALL Java_com_ndkdemo_MathKit_square
            (JNIEnv *env, jclass cls, jint num){
        return num * num;
    }
    

    在app module目录下的build.gradle配置ndk选项

    defaultConfig {
        ......
        ndk{
               moduleName "ndklib"         //生成的so名字,实际为 libndklib.so
               abiFilters "armeabi", "armeabi-v7a", "arm64-v8a", "x86"  //输出指定三种abi体系结构下的so库
            }
    }
    buildTypes {
            release {
                minifyEnabled false
                proguardFiles 'proguard-rules.pro'
                ndk {
                    moduleName "jnimain"
                    abiFilters "armeabi", "armeabi-v7a"
                }
            }
        }
    

    Make上述Library项目,生成so文件

    先在gradle.properties中添加:android.useDeprecatedNdk=true

    生成的so文件在如下目录,生成的so文件为 =lib+ 配置生成名 .so
    <Module主目录>/build/intermediates/ndk/debug/lib/arm64-v8a/libndklib.so
    <Module主目录>/build/intermediates/ndk/debug/lib/armeabi/libndklib.so
    <Module主目录>/build/intermediates/ndk/debug/lib/armeabi-v7a/libndklib.so
    <Module主目录>/build/intermediates/ndk/debug/lib/x86/libndklib.so

    在其他Application项目中引用

    1.直接引用Library Module
    定义和模板类相同类,包名+类名和此前jni中一致

    // 包名和类名要和.cpp或者.c文件中一致
    package com.ndkdemo;
    
    public class MathKit{
        static{
            // 对应库文件名称,要一致。生成的.so文件名为libndklib.so,
            // 那么loadLibrary为ndklib,去掉前面的lib及后面的.so
            System.loadLibrary("NDKDemo");
        }
       // 定义native本地方法,和普通方法相同,加上native关键字
        public static native int square(int num);
    }
    

    在应用中使用静态方式调用native方法
    例如: MathKit.square(10)

    2.创建src/main/jniLibs目录,把生成的so文件拷贝进去调用natvie方法方式同上

    自定义jni路径和so文件路径

    1.jni编辑路径自定义

    android {
    
      sourceSets.main {
          jni.srcDirs 'src/main/source'
      }
    }
    

    2.so文件路径自定义

    sourceSets {
            main {
                jniLibs.srcDirs = ['libs']
            }
        }
    

    由于Android Studio以强大的方式集成了NDK, 所以上面很多配置都不需要写. 方便了很多..mk文件不用自己写。

    相关文章

      网友评论

      • aa38da6e413b:请教一下,我安装完环境之后c++代码无补全和提示功能,也无法点击jni.h进行查看,是什么原因。代码可以编译通过

      本文标题:Android开发之NDK

      本文链接:https://www.haomeiwen.com/subject/zubeattx.html