如何将本机代码库嵌入Android应用程序SDK 29+中?

如何解决如何将本机代码库嵌入Android应用程序SDK 29+中?

我有一个使用本机代码库的应用程序-ffmpeg的自定义构建,尽管这并不严格相关。

到目前为止,我已将可执行文件添加到我的res / raw目录中,然后在运行时将其提取到数据文件夹中,然后使用exec()对其进行调用。但是,由于compileSdkVersion = 29,由于W ^ X漏洞的新安全性,它已停止工作。现在,调用exec()会导致异常:

java.io.IOException: Cannot run program "..." (in directory "..."): error=13,Permission denied

我已阅读以下页面:
permission is denied using Android Q ffmpeg": error=13,Permission denied
https://issuetracker.google.com/issues/152645643

据我所知,正确的方法是:

  1. 将我的库放在项目的/ libs目录中
  2. android:extractNativeLibs = "true"添加到AndroidManifest.xml
  3. 确保文件名匹配模式“ lib * .so”

然后安装程序应该将库解压缩到可执行位置,并且我应该能够在运行时读取它。

这只是不起作用。我一定错过了重要的一步。

我的代码

app / build.gradle:

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'

android {
    compileSdkVersion 30
    buildToolsVersion "30.0.1"

    defaultConfig {
        applicationId "com.example.nativelibrarytest"
        minSdkVersion 29
        targetSdkVersion 30
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'),'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(dir: "libs",include: ["*.jar"])
    implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
    implementation 'androidx.core:core-ktx:1.3.1'
    implementation 'androidx.appcompat:appcompat:1.2.0'
    implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.2'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
}

AndroidManifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.nativelibrarytest">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:extractNativeLibs="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

活动代码(基于基本活动模板):

class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
    }

    override fun onResume() {
        super.onResume()

        // Test code: see if we can find the /libs library
        val libName = "libffmpeg.so"
        val nativeLibsDir = applicationInfo.nativeLibraryDir
        val libPath = "$nativeLibsDir/$libName"
        val libFile = File(libPath)
        val result = if (libFile.exists()) "exists" else "does not exist"

        val label = findViewById<TextView>(R.id.label)
        label.text = "$libPath $result"

        Log.d("MainActivity",label.text.toString())
    }
}

项目树

enter image description here

结果(在模拟器上)

/data/app/~~Q0jHO-XwEvieuwr-92_Fyg==/com.example.nativelibrarytest-byqCoqnbYsgtmCOVg361NA==/lib/x86/libffmpeg.so does not exist

为什么我的运行时代码找不到库?

我尝试过:

  • 在build.gradle>依赖项{...}
  • 中添加implementation files('libs/libffmpeg.so')
  • 添加implementation fileTree(dir: "libs",include: ["lib*.so"])
  • 创建libs / x86和libs / arm64子目录并将libffmpeg.so放在这些子目录中

我已经扩展了.apk文件并将其解压缩,并且其中没有lib目录。好像构建过程没有意识到本机库存在并且没有将其复制到其中。

解决方法

更新:经过大量的Google搜索,我认为我已经找到了答案。我的build.gradle文件中缺少ndk abiFilterssourceSets

所以我的最终项目设置是:

  1. 在模块的根目录下创建一个libs文件夹,其中包含每种体系结构的子文件夹。每个二进制文件都必须严格命名为lib<something>.so,即使它不是严格的共享库。
|app/
|-- build/
|-- libs/
|---- arm64-v8a/
|------ libffmpeg.so (compiled for 64-bit ARM)
|---- armeabi-v7a/
|------ libffmpeg.so (compiled for 32-bit ARM)
|---- x86/
|------ libffmpeg.so (compiled for 32-bit Intel)
|---- x86_64/
|------ libffmpeg.so (compiled for 64-bit Intel)
|-- src/

Final project layout

    该应用程序的
  1. build.gradle应包含以下条目
  • android> defaultConfig> ndk> abiFilters
  • android> sourceSets
  • 依赖关系>实现fileTree

这是完整的build.gradle:

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'

android {
    compileSdkVersion 30
    buildToolsVersion "30.0.1"

    defaultConfig {
        applicationId "com.example.nativelibrarytest"
        minSdkVersion 29
        targetSdkVersion 30
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"

        ndk {
            abiFilters 'armeabi-v7a','arm64-v8a','x86','x86_64'
        }
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'),'proguard-rules.pro'
        }
    }

    sourceSets {
        main {
            // let gradle pack the shared library into apk
            jniLibs.srcDirs = ['libs']
        }
    }
}

dependencies {
    implementation fileTree(dir: "libs",include: ["lib*.so"])
    implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
    implementation 'androidx.core:core-ktx:1.3.1'
    implementation 'androidx.appcompat:appcompat:1.2.0'
    implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.2'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
}

AndroidManifest.xml和活动代码与我的原始帖子相同。

Logcat结果是:

2020-09-29 19:32:50.911 9866-9866/com.example.nativelibrarytest D/MainActivity: /data/app/~~zwvTGRwVjL_Uv5d4joi4jw==/com.example.nativelibrarytest-9G5w819CD5uenr5N1jiAhg==/lib/x86/libffmpeg.so exists

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-