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 更新实现。
相关推荐
IT_陈寒13 小时前
React状态管理终极对决:Redux vs Context API谁更胜一筹?
前端·人工智能·后端
Kagol14 小时前
TinyVue 支持 Skills 啦!现在你可以让 AI 使用 TinyVue 组件搭建项目
前端·agent·ai编程
柳杉14 小时前
从零打造 AI 全球趋势监测大屏
前端·javascript·aigc
simple_lau14 小时前
Cursor配置MasterGo MCP:一键读取设计稿生成高还原度前端代码
前端·javascript·vue.js
睡不着先生14 小时前
如何设计一个真正可扩展的表单生成器?
前端·javascript·vue.js
天蓝色的鱼鱼14 小时前
模块化与组件化:90%的前端开发者都没搞懂的本质区别
前端·架构·代码规范
明君8799714 小时前
Flutter 如何给图片添加多行文字水印
前端·flutter
leolee1815 小时前
Redux Toolkit 实战使用指南
前端·react.js·redux
bluceli15 小时前
React Hooks最佳实践:写出优雅高效的组件代码
前端·react.js
IT_陈寒15 小时前
JavaScript代码效率提升50%?这5个优化技巧你必须知道!
前端·人工智能·后端