Flutter 项目热更新加载 libapp.so 文件

Flutter 项目热更新加载 libapp.so 文件


1. 获取 .so 文件

  1. 打包 Flutter 项目的 Release 版 APK。
  2. 解压 APK 文件,从 lib/arm64-v8alib/armeabi-v7a 目录下提取 libapp.so 文件。

2. 加载 .so 文件的实现代码

libapp.so 文件放入 Android 项目的 app/src/main/assets 目录下,并通过以下代码下载到应用私有目录中:

java

arduino 复制代码
private static final String LIB_NAME = "libapp.so";

public void loadSoFromAssets() {
    try {
        // 使用应用私有目录存储 .so 文件
        File libDir = new File(getFilesDir(), "libs");
        if (!libDir.exists()) {
            libDir.mkdirs();
        }
        File libFile = new File(libDir, LIB_NAME);

        // 检查文件是否已存在
        if (libFile.exists()) {
            Log.d("LoadSO", "SO 文件已存在,路径: " + libFile.getAbsolutePath());
            Log.d("LoadSO", "文件大小: " + libFile.length() + " bytes");
            Log.d("LoadSO", "最后修改时间: " + new Date(libFile.lastModified()));
            Log.d("LoadSO", "是否可读: " + libFile.canRead());
            Log.d("LoadSO", "是否可写: " + libFile.canWrite());

            System.load(libFile.getAbsolutePath());
            Toast.makeText(this, "加载已存在的 SO 文件: " + libFile.getAbsolutePath(), Toast.LENGTH_SHORT).show();
            return;
        }

        // 从 assets 复制到私有目录
        InputStream is = getAssets().open(LIB_NAME);
        FileOutputStream fos = new FileOutputStream(libFile);
        byte[] buffer = new byte[1024];
        int len;
        while ((len = is.read(buffer)) > 0) {
            fos.write(buffer, 0, len);
        }
        fos.close();
        is.close();

        Log.d("LoadSO", "SO 文件加载成功,路径: " + libFile.getAbsolutePath());
        Log.d("LoadSO", "文件大小: " + libFile.length() + " bytes");
        Log.d("LoadSO", "最后修改时间: " + new Date(libFile.lastModified()));
        Log.d("LoadSO", "是否可读: " + libFile.canRead());
        Log.d("LoadSO", "是否可写: " + libFile.canWrite());

        // 加载 .so 文件
        System.load(libFile.getAbsolutePath());
        Toast.makeText(this, "加载 SO 文件成功: " + libFile.getAbsolutePath(), Toast.LENGTH_SHORT).show();
    } catch (Exception e) {
        Toast.makeText(this, "加载 SO 文件失败: " + e.getMessage(), Toast.LENGTH_SHORT).show();
        Log.e("LoadSO", "加载 SO 文件失败", e);
    }
}

3. 测试结果

  • libapp.so 文件下载到本地目录后,调用 System.load(libFile.getAbsolutePath()) 加载。
  • 重新进入应用时,UI 未更新到预期的变化。

4. 结论

  1. 限制 1:虽然 .so 文件被复制到了应用私有目录,但 Android 系统不允许在运行时热替换已加载的原生库。
  2. 限制 2:Android 对 .so 文件的加载机制有严格限制,一旦加载后无法在运行时替换。

5. 注意事项

  • 动态加载 .so 文件需谨慎,确保文件路径和权限正确。
  • 若需更新原生代码,建议通过完整的 APK 更新实现。
相关推荐
fanruitian4 小时前
uniapp android开发 测试板本与发行版本
前端·javascript·uni-app
rayufo4 小时前
【工具】列出指定文件夹下所有的目录和文件
开发语言·前端·python
RANCE_atttackkk4 小时前
[Java]实现使用邮箱找回密码的功能
java·开发语言·前端·spring boot·intellij-idea·idea
2501_944525545 小时前
Flutter for OpenHarmony 个人理财管理App实战 - 支出分析页面
android·开发语言·前端·javascript·flutter
李白你好6 小时前
Burp Suite插件用于自动检测Web应用程序中的未授权访问漏洞
前端
刘一说7 小时前
Vue 组件不必要的重新渲染问题解析:为什么子组件总在“无故”刷新?
前端·javascript·vue.js
徐同保8 小时前
React useRef 完全指南:在异步回调中访问最新的 props/state引言
前端·javascript·react.js
刘一说8 小时前
Vue 导航守卫未生效问题解析:为什么路由守卫不执行或逻辑失效?
前端·javascript·vue.js
一周七喜h9 小时前
在Vue3和TypeScripts中使用pinia
前端·javascript·vue.js
weixin_395448919 小时前
main.c_cursor_0202
前端·网络·算法