Flutter 热更新功能实现

initAot(applicationContext);
initResources(applicationContext);
System.loadLibrary(“flutter”);
long initTimeMillis = SystemClock.uptimeMillis() - initStartTimestampMillis;
nativeRecordStartTimestamp(initTimeMillis);
}
}

在 startInitialization 中,主要执行了三个初始化方法 initConfig(applicationContext),initAot(applicationContext),initResources(applicationContext),最后记录了执行时间;

在 initConfig 中:

private static void initConfig(Context applicationContext) {
try {
Bundle metadata = applicationContext.getPackageManager().getApplicationInfo(applicationContext.getPackageName(), 128).metaData;
if (metadata != null) {
sAotSharedLibraryPath = metadata.getString(PUBLIC_AOT_AOT_SHARED_LIBRARY_PATH, “app.so”);
sAotVmSnapshotData = metadata.getString(PUBLIC_AOT_VM_SNAPSHOT_DATA_KEY, “vm_snapshot_data”);
sAotVmSnapshotInstr = metadata.getString(PUBLIC_AOT_VM_SNAPSHOT_INSTR_KEY, “vm_snapshot_instr”);
sAotIsolateSnapshotData = metadata.getString(PUBLIC_AOT_ISOLATE_SNAPSHOT_DATA_KEY, “isolate_snapshot_data”);
sAotIsolateSnapshotInstr = metadata.getString(PUBLIC_AOT_ISOLATE_SNAPSHOT_INSTR_KEY, “isolate_snapshot_instr”);
sFlx = metadata.getString(PUBLIC_FLX_KEY, “app.flx”);
sFlutterAssetsDir = metadata.getString(PUBLIC_FLUTTER_ASSETS_DIR_KEY, “flutter_assets”);
}

} catch (NameNotFoundException var2) {
throw new RuntimeException(var2);
}
}

在 initResources 中:

sResourceExtractor = new ResourceExtractor(applicationContext);
sResourceExtractor.addResource(fromFlutterAssets(sFlx)).addResource(fromFlutterAssets(sAotVmSnapshotData)).addResource(fromFlutterAssets(sAotVmSnapshotInstr)).addResource(fromFlutterAssets(sAotIsolateSnapshotData)).addResource(fromFlutterAssets(sAotIsolateSnapshotInstr)).addResource(fromFlutterAssets(“kernel_blob.bin”));
if (sIsPrecompiledAsSharedLibrary) {
sResourceExtractor.addResource(sAotSharedLibraryPath);
} else {
sResourceExtractor.addResource(sAotVmSnapshotData).addResource(sAotVmSnapshotInstr).addResource(sAotIsolateSnapshotData).addResource(sAotIsolateSnapshotInstr);
}

sResourceExtractor.start();

在 ResourceExtractor 类中,通过名字就能知道这个类是做资源提取的。把 add 的 Flutter 相关文件从 assets 目录中取出来,该类中 ExtractTask 的 doInBackground 方法中:

File dataDir = new File(PathUtils.getDataDirectory(ResourceExtractor.this.mContext));

这句话指定了资源提取的目的地,即 data/data/包名/app_flutter,如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

如图,可以看到该目录是的访问权限是可读可写,所以理论上,我们只要把自己的 Flutter 产物下载后,从内存 copy 到这里,便能够实现代码的动态更新。

代码实现

public class FlutterUtils { private static String TAG = “FlutterUtils.class”; private static String flutterZipName = “flutter-code.zip”; private static String fileSuffix = “.zip”; private static String zipPath = Environment.getExternalStorageDirectory().getPath() + “/k12/” + flutterZipName; private static String targetDirPath = zipPath.replace(fileSuffix, “”); private static String targetDirDataPath = zipPath.replace(fileSuffix, “/data”); /** * Flutter 代码热更新第一步: 解压 Flutter 的压缩文件 / public static void unZipFlutterFile() { Log.i(TAG, “unZipFile: Start”); try { unZipFile(zipPath, targetDirPath); Log.i(TAG, “unZipFile: Finish”); } catch (Exception e) { e.printStackTrace(); } } /* * Flutter 代码热更新第二步: 将 Flutter 的相关文件移动到 AppData 的相关目录,APP启动时调用 * * @param mContext 获取 AppData 目录需要 / public static void copyDataToFlutterAssets(Context mContext) { String appDataDirPath = PathUtils.getDataDirectory(mContext.getApplicationContext()) + File.separator; Log.d(TAG, “copyDataToFlutterAssets-filesDirPath:” + targetDirDataPath); Log.d(TAG, “copyDataToFlutterAssets-appDataDirPath:” + appDataDirPath); File appDataDirFile = new File(appDataDirPath); File filesDirFile = new File(targetDirDataPath); File[] files = filesDirFile.listFiles(); for (File srcFile : files) { if (srcFile.getPath().contains(“isolate_snapshot_data”) || srcFile.getPath().contains(“isolate_snapshot_instr”) || srcFile.getPath().contains(“vm_snapshot_data”) || srcFile.getPath().contains(“vm_snapshot_instr”)) { File targetFile = new File(appDataDirFile + “/” + srcFile.getName()); FileUtil.copyFileByFileChannels(srcFile, targetFile); Log.i(TAG, “copyDataToFlutterAssets-copyFile:” + srcFile.getPath()); } } Log.i(TAG, “copyDataToFlutterAssets: Finish”); } /* * 解压缩文件到指定目录 * * @param zipFileString 压缩文件路径 * @param outPathString 目标路径 * @throws Exception / private static void unZipFile(String zipFileString, String outPathString) { try { ZipInputStream inZip = new ZipInputStream(new FileInputStream(zipFileString)); ZipEntry zipEntry; String szName = “”; while ((zipEntry = inZip.getNextEntry()) != null) { szName = zipEntry.getName(); if (zipEntry.isDirectory()) { szName = szName.substring(0, szName.length() - 1); File folder = new File(outPathString + File.separator + szName); folder.mkdirs(); } else { File file = new File(outPathString + File.separator + szName); if (!file.exists()) { Log.d(TAG, “Create the file:” + outPathString + File.separator + szName); file.getParentFile().mkdirs(); file.createNewFile(); } FileOutputStream out = new FileOutputStream(file); int len; byte[] buffer = new byte[1024]; while ((len = inZip.read(buffer)) != -1) { out.write(buffer, 0, len); out.flush(); } out.close(); } } inZip.close(); } catch (Exception e) { Log.i(TAG,e.getMessage()); e.printStackTrace(); } } /* * 使用FileChannels复制文件。 * * @param source 原路径 * @param dest 目标路径 / public static void copyFileByFileChannels(File source, File dest) { FileChannel inputChannel = null; FileChannel outputChannel = null; try { inputChannel = new FileInputStream(source).getChannel(); outputChannel = new FileOutputStream(dest).getChannel(); outputChannel.transferFrom(inputChannel, 0, inputChannel.size()); refreshMedia(BaseApplication.getBaseApplication(), dest); } catch (Exception e) { e.printStackTrace(); } finally { try { inputChannel.close(); outputChannel.close(); } catch (IOException e) { e.printStackTrace(); } } } /* * 更新媒体库 * * @param cxt * @param files */ public static void refreshMedia(Context cxt, File… files) { for (File file : files) { String filePath = file.getAbsolutePath(); refreshMedia(cxt, filePath); } } public static void refreshMedia(Context cxt, String… filePaths) { MediaScannerConnection.scanFile(cxt.getApplicationContext(), filePaths, null, null); }}

Flutter 资源的热更新

我们的App安装到手机上后,是很难再修改 Assets 目录下的资源,所以关于资源的替换,目前的方案是使用 Flutter 的 API :Image.file() 来从存储卡中读取图片。

通常我们的 Flutter 项目中应当存有关于 App 的图片,尽量保证在热更新的时候使用已经存在的图片,

其次,我们可以使用 Image.network() 来加载网络资源的图片,

如果还不能满足需求,兜底的方案就是使用 Image.file(),将资源图片放到Zip目录下一起下发,并在Flutter代码中使用 Image.file() 来加载。

  1. 通过 Native 层方法拿到图片文件夹的内存地址 dataDir;
  2. 判断图片是否存在,存在则加载,不存在则加载已经存在的图片占位;

new File(dataDir + ‘hotupdate_test.png’).existsSync()

? Image.file(new File(dataDir + ‘hotupdate_test.png’))

: Image.asset(“images/net_error.png”),

总结

在 Flutter 代码产物替换中,因为替换的 4 个文件皆为直接加载到内存中的引擎代码,所以这部分优化空间有限。但在资源的热更新中,资源是从Assets取得,所以这里应该有更优的方案。
Flutter 的热更新意味着可以在在App的一个入口里,像 H5 一样无穷的嵌入页面,但又有和原生媲美的流畅体验。
未来 Flutter 热更新技术如果成熟,应用开发可能只需要 Android端和 IOS端实现本地业务功能模块的封装,业务和UI的代码都放在 Flutter 中,
便能够真正的实现移动两端一份业务代码,并且赋予产品在不影响用户体验的情况下,拥有动态部署APP内容的能力。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

总结

最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2019-2021面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节

还有 高级架构技术进阶脑图、Android开发面试专题资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

一线互联网面试专题

379页的Android进阶知识大全

379页的Android进阶知识大全

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。祝大家2021年万事大吉。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

自己,没有做不到的,只有想不到的。祝大家2021年万事大吉。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 7
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Flutter热更新通常是指在运行时更新 Flutter 应用程序的代码和资源,而无需重新安装应用程序。Flutter 支持热更新的原因是因为它是一种跨平台框架,并且具有热重载功能。本文将介绍如何实现 Flutter 应用程序的热更新热更新实现原理是,将新的代码和资源文件下载到本地存储,并使用 Flutter 的热重载功能重新加载这些文件。具体步骤如下: 1.使用 Flutter 的热重载功能Flutter 中,热重载是一种在运行时重新加载应用程序代码的功能,可以大大加快开发速度。我们可以通过在 Flutter IDE 中按下“r”键或运行 flutter run 命令来启用热重载功能。 但是,热重载只能更新 Dart 代码,无法更新原生代码。因此,我们需要在 Dart 代码中实现下载和更新原生代码的逻辑。 2.下载新的代码和资源文件 我们可以使用 Dart 的 http 包或 dio 包来下载新的代码和资源文件。下载的文件可以存储在设备的本地存储中,例如使用 Flutter 的 path_provider 包来获取本地存储路径。 3.更新代码和资源文件 在下载完新的代码和资源文件后,我们需要使用 Flutter 的热重载功能重新加载这些文件。我们可以通过在 Dart 代码中调用 Flutter 的 reload 方法来实现: ``` await FlutterReloader.reload(); ``` 这里的 FlutterReloader 是一个第三方库,用于封装 Flutter 的热重载功能。我们可以在 pubspec.yaml 文件中添加以下依赖: ``` dependencies: flutter_reloader: ^1.2.0 ``` 4.处理错误 在更新代码和资源文件时,可能会出现一些错误,例如下载失败、解压失败等。我们需要在代码中处理这些错误,并回滚到之前的版本。 综上所述,实现 Flutter 应用程序的热更新需要使用 Dart 的 http 包或 dio 包下载新的代码和资源文件,并使用 Flutter 的热重载功能重新加载这些文件。在实际应用中,我们还需要处理错误和回滚到之前的版本等问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值