使用 android gradle 实验链接 .aar 文件中的 .so 文件

问题描述 投票:0回答:1

我正在使用 android gradle 实验插件使用一些本机代码构建应用程序模块。此本机代码使用带有预构建 .so 文件的库,我通过 android 库模块将其捆绑到 .aar 文件中。 .aar 文件构建得很好,但如何将应用程序模块中的本机代码模块链接到 .aar 模块中预构建的 .so 文件? gradle实验文档没有提到这种场景。

另外,如果可能的话,我想将包含文件打包到 .aar 文件中(尽管它们不应该与最终应用程序一起打包)。

在 /prebuiltlib 中:

build.gradle
-src/
--main/
---jniLibs/
----libfoo.so

这是 gradle 文件:

/prebuiltlib/build.gradle

apply plugin: "com.android.model.library"
model {
    android {
        compileSdkVersion 25
        buildToolsVersion "25.0.3"

        defaultConfig {
            minSdkVersion.apiLevel = 21
            targetSdkVersion.apiLevel = 21
            versionCode 1
            versionName "1.0"

        }

        buildTypes {
            release {
                minifyEnabled false
                proguardFiles.add(file("proguard-rules.pro"))
            }
        }
    }
}

dependencies {
    compile fileTree(dir: "libs", include: ["*.jar"])
    compile "com.android.support:appcompat-v7:25.3.1"
}

这里是/app/build.gradle,注意???我不知道该放什么:

import org.apache.tools.ant.taskdefs.condition.Os
apply plugin: 'com.android.model.application'

model {
    repositories {
        libs(PrebuiltLibraries) {
            // ??? is this right, and does this go into app/build.gradle or mylib/build.gradle?
            foo {
                binaries.withType(SharedLibraryBinary) {
                    sharedLibraryFile = file('???/libfoo.so')
                }
            }
        }
    }
    android {
        compileSdkVersion = 25
        buildToolsVersion = '25.0.3'

        defaultConfig {
            applicationId = 'com.jeremy.stackoverflow.sample'
            minSdkVersion.apiLevel = 21
            targetSdkVersion.apiLevel = 21
            versionCode = 1
            versionName = '1.0'
        }

        ndk {
            platformVersion = 21
            moduleName = 'native-activity'
            toolchain = 'gcc'
            toolchainVersion = '4.9'
            stl = 'gnustl_shared'
            abiFilters.add('armeabi-v7a')
            ldLibs.addAll(['log',
                           'android',
                           'EGL',
                           'GLESv2'
            ])
            // ??? How do I add include paths from the .aar module?
            cppFlags.add('-I' + file('???/include'))
            cppFlags.addAll(['-std=c++11', '-fexceptions'])
        }

        sources {
            main {
                jni {
                    dependencies {
                        // ??? Is this right?
                        library 'foo' linkage 'shared'
                    }
                }
                jniLibs {
                    source {
                        // ??? Do I need one of these for the libs in the .aar?
                        srcDir file('???/jniLibs')
                    }
                    dependencies {
                        // ??? Is this right?
                        library 'foo'
                    }
                }
            }
        }
        buildTypes {
            release {
                minifyEnabled = false
                proguardFiles.add(file('proguard-rules.pro'))
            }
        }
    }
}

dependencies {
    println rootProject.getName()
    compile fileTree(include: ['*.jar'], dir: 'libs')
    compile 'com.android.support:appcompat-v7:25.3.1'
    compile project(':prebuiltlib')
}
android gradle android-ndk gradle-experimental
1个回答
0
投票

从 Android Gradle Plugin 4.0 开始,可以从 build.gradle 文件中链接的 AAR 导入 C/C++ 依赖项。 Gradle 会自动使这些可用于本机构建系统,但您的构建系统必须配置为使用导入的库和标头。

在导入模块的应用程序的build.gradle上添加:

在 gradle 4.0 中:将以下内容添加到项目的 gradle.properties 文件中:

android.enablePrefab=true

在 gradle 4.1 中:将以下内容添加到模块的 build.gradle 文件的 android 块中:

buildFeatures {
  prefab true
}

然后添加对必须导入的模块的依赖关系:

dependencies {
    implementation files('libs/nativelib-release.aar')
}

在本例中,发布版本中的 lib aar 被放置在 app/libs 目录中。

在要导入的模块的build.gradle上添加:

buildFeatures {
    prefabPublishing = true
}

prefab {
    create("nativelib") {
      headers = "src/main/cpp/include"
    }

    // if you have other libs to expose...
    create("myotherlibrary") {
        headers = "src/main/cpp/myotherlibrary/include"
    } 
}

请注意,目前仅允许使用一个包含路径(必须放置包含公共 API 的 .h),并且此处引发了一个错误:https://issuetracker.google.com/issues/168775349

,如果您的应用程序定义了 libapp.so 并且使用了 cURL,则您的 CMakeLists.txt 应包含以下内容:

add_library(app SHARED app.cpp)

# Add these two lines.
find_package(nativelib REQUIRED CONFIG)
target_link_libraries(app nativelib::nativelib)

app.cpp 现在能够#include“curl/curl.h”,libapp.so 在构建时将自动链接到 libcurl.so,并且 libcurl.so 将包含在 APK 中。

来源:https://developer.android.com/studio/build/native-dependency

额外位:

可能需要使用共享的 stl 构建 aar 模块,您可以在模块 gradle 文件的 android defaultConfig 上添加以下内容:

externalNativeBuild {
    cmake {
        cppFlags ""
        arguments "-DANDROID_STL=c++_shared"
    }
}

用于调试:

模块库将通过变量

CMAKE_FIND_ROOT_PATH
暴露给应用程序的cmake。使用
message(${CMAKE_FIND_ROOT_PATH})
显示该路径,里面会有一个名为
nativelibConfig.cmake
的文件(或另一个库/模块名称)。 该文件包含以下内容:

if(NOT TARGET nativelib::nativelib)
add_library(nativelib::nativelib SHARED IMPORTED)
set_target_properties(nativelib::nativelib PROPERTIES
    IMPORTED_LOCATION "/Users/bloom/.gradle/caches/transforms-3/44ca3477885179ff4cb5073527c9d262/transformed/nativelib-debug/prefab/modules/nativelib/libs/android.arm64-v8a/libnativelib.so"
    INTERFACE_INCLUDE_DIRECTORIES "/Users/bloom/.gradle/caches/transforms-3/44ca3477885179ff4cb5073527c9d262/transformed/nativelib-debug/prefab/modules/nativelib/include"
    INTERFACE_LINK_LIBRARIES ""
)
endif()

它以 :: 格式显示库名称(非常奇怪),以防很难猜测在 CMakelists.txt 上使用的名称和导出的路径。

要刷新编辑器等,请尝试构建->刷新链接的 C++ 项目。

© www.soinside.com 2019 - 2024. All rights reserved.