android SDK开发之Native 依赖

作者: 豪_77e3 | 来源:发表于2019-08-26 10:39 被阅读0次
    1. 2个独立的SDK之间难免会有native依赖,例如A SDK 依赖 B SDK,我们通常的做法是手动将A中的头文件和已经编译好的lib拷贝到B中,然后在B中的Android.mk或是CMakeList.txt中声明好依赖。
    2. 这个方法的缺点是:
      • A和B不止有native依赖,也有java间的依赖,将native和java依赖分开更新,很容易出现问题。
      • A的头文件和native库每次都要提交到B中的svn/git中,由于native库比较大,速度会很慢,也不利于版本匹配, 当出现问题时,很难定位到依赖A的哪个版本
    3. 新方法的要求:
      • android studio IDE
      • android gradle tool 插件版本:3.0.0 - 3.3.0
    4. 新方法的使用:
      • 打开SDK根目录的build.gradle,增加对nativeBundel插件的依赖,如下
    buildscript {
      repositories {
          jcenter()
          google()
      }
      dependencies {
          // android gradle tool plugin
          classpath 'com.android.tools.build:gradle:3.0.0'
          //Add nativeBundle plugin dependency
          classpath 'com.ydq.android.gradle.build.tool:nativeBundle:1.0.1'
      }
    }
    
    • 对于被依赖者A来说

      1. 打开A 模块的build.gradle,引用nativeBundel的导出插件
      apply plugin: 'com.android.library'
      apply plugin: 'maven-publish'
      apply plugin: 'com.ydq.android.gradle.native-aar.export' // must below android gradle plugin
      
      1. 设置导出头文件,在build.gradle,中增加下面设置,头文件会打包进AAR包里。
      nativeBundleExport {
        headerDir = "${project.projectDir}/src/main/jni/include"
      }
      
      1. 支持flavor,如果你的工程有使用的话
      productFlavors {
        flavorDimensions "default"
        export {
            dimension "default"
            nativeBundleExport {
                headerDir = "${project.projectDir}/src/main/jni/include"
            }
        }
      }
      
      1. 支持发布静态库AAR,如果设置了bundleStatic,插件会生个bundleStaticLibDebug/Rlease任务来打包静态库,你可以依赖这个任务的输出AAR作为静态版本,这个AAR的输出目录为${project.buildDir}/output/aar/,如果是release编译,名字为${project.name}-static-release.aar
      nativeBundleExport {
          headerDir = "${project.projectDir}/src/main/jni/include"
          bundleStatic = true
          //extraStaticLibDir = "${project.projectDir}/xx"
          //excludeStaticLibs.add("**/libmylib.a")
          //excludeStaticLibs.add("**/libxx.a")
          //linkOrder = "libxx.a:libyy.a"
      }
      

      5.对于静态版本,你可以指定额外的静态库目录extraStaticLibDir,默认会收集gradle externalNativeBuild的输出静态库;其次你可以排除一些静态库,通过excludeStaticLibs来指定。由于发布多个静态库,可能会涉及到链接顺序问题在链接成动态库时,如果多个静态库间有依赖的话。你可以指定链接顺序。

    • 对于使用者B来说

      1. 打开A 模块的build.gradle,引用nativeBundel的导入插件
      apply plugin: 'com.android.library'
      apply plugin: 'maven-publish'
      apply plugin: 'com.ydq.android.gradle.native-aar.import' // must below android gradle plugin
      
      
      1. 如果你是使用externalNativeBuild ndkBuild来配置native编译的话,打开Android.mk,将include ${ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK} #must followed by "include $(BUILD_SHARED_LIBRARY)" or "include $(BUILD_STATIC_LIBRARY)" 放在需要依赖A库模块的 include $(BUILD_SHARED_LIBRARY)或者include $(BUILD_STATIC_LIBRARY)之前,例如
      include $(CLEAR_VARS)
      LOCAL_SRC_FILES := myapp.cpp \
      LOCAL_MODULE := myapp
      LOCAL_LDLIBS += -llog
      include ${ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK} #must followed by "include $(BUILD_SHARED_LIBRARY)" or "include   $(BUILD_STATIC_LIBRARY)"
      include $(BUILD_SHARED_LIBRARY)
      
      

      3.如果你是使用externalNativeBuild cmake来配置native编译的话,将include (${ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK}) 添加到CMakeList.txt,然后在需要依赖A库的target,加上target_link_libraries(<target> ${ANDROID_GRADLE_NATIVE_MODULES});例如

      cmake_minimum_required(VERSION 3.4.1)
      project(echo LANGUAGES C CXX)
      add_library(myapp
        SHARED
        myapp.cpp)
      target_link_libraries(myapp
        log
        )
      include (${ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK})
      target_link_libraries(myapp ${ANDROID_GRADLE_NATIVE_MODULES})
      target_compile_options(myapp
        PRIVATE
        -Wall -Werror)
      
      1. 如果你是用自定义task NDK-BUILD的话,将ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK=${nativeBundleImport.ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK}增加到你的命令中,例如
      def execmd = ["$ndkbuildcmd", "-j${coreNum}", "V=1", "NDK_PORJECT_PATH=$buildDir",
                              "APP_BUILD_SCRIPT=$androidMKfile", "NDK_APPLICATION_MK=$applicationMKfile", "ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK=${nativeBundleImport.ANDROID_GRADLE_NATIVE_BUNDLE_PLUGIN_MK}"]
      
      1. 如果你想链接整个静态库的话,可以像下面声明
      nativeBundleImport {
        wholeStaticLibs = "libxx.a:libyy.a" // Library is seperated by colon
      }
      
      1. 如果你的模块有直接对so的依赖,例如下面的样子,这个插件会将so打包进这个库的aar里,而且会将它放入native构建里
      dependencies {
            implementation "com.my.group:module:1.0.0:armeabi-v7a@so"
            implementation "com.my.group:module:1.0.0:armeabi-v7a@har" // contain 'headers'
        }
      
      1. 如果你不想将某个依赖放进native构建里,例如:implementation "com.my.group:moduleA:1.0.0" 有native接口(包括了头文件和so),你可以这样做
      nativeBundleImport {
            //wholeStaticLibs = "libxx.a:libyy.a" // Library is seperated by colon
            excludeDependencies.add("com.my.group:moduleA")
            excludeDependencies.add("com.my.group:moduleB")
        }
      
      1. 如果上面的方法都无法满足你的需求,你可以到${project.projectDir}/build/nativeLib/目录下查找头文件和native库,自己声明如何使用。

    github:https://github.com/howardpang/androidNativeBundle

    相关文章

      网友评论

        本文标题:android SDK开发之Native 依赖

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