【译】 Flutter 延迟加载组件 【包体积优化 _ 动态化】

本文介绍了如何在Flutter应用中使用SplitCompat和DeferredComponents进行模块化加载,包括在attachBaseContext中安装SplitCompat,声明SplitCompatApplication,处理延迟组件的安装,以及如何编写和使用延迟加载的Dart库,如DeferredBox。指南还涵盖了构建过程和验证步骤,以确保组件的正确加载和配置。
摘要由CSDN通过智能技术生成


}

  • 在 `attachBaseContext()` 中调用 `SplitCompat.install(this);`:

@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
// Emulates installation of future on demand modules using SplitCompat.
SplitCompat.install(this);
}

  • 将 `SplitCompatApplication` 声明为 application 的子类, 并将 `FlutterApplication` 中的 flutter 兼容性代码添加到你的 application 类中:

<application

android:name=“com.google.android.play.core.splitcompat.SplitCompatApplication”>

嵌入层依赖注入的 DeferredComponentManager 实例来处理延迟组件的安装请求。 通过在应用程序的初始流程中添加以下代码,将 PlayStoreDeferredComponentManager 添加到 Flutter 嵌入层中:

import io.flutter.embedding.engine.dynamicfeatures.PlayStoreDeferredComponentManager;
import io.flutter.FlutterInjector;

layStoreDeferredComponentManager deferredComponentManager = new
PlayStoreDeferredComponentManager(this, null);
FlutterInjector.setInstance(new FlutterInjector.Builder()
.setDeferredComponentManager(deferredComponentManager).build());

  1. 通过将 `deferred-components` 依赖添加到应用程序的 `pubspec.yaml` 中的 `flutter` 下,并选择延迟组件:


flutter:

deferred-components:

flutter 工具会在 pubspec.yaml 中查找 deferred-components, 来确定是否应将应用程序构建为延迟加载。 除非你已经知道所需的组件和每个组件中的 Dart 延迟库,否则可以暂时将其留空。 当 gen_snapshot 生成加载单元后,你可以在后面的 步骤 3.3 中完善这部分内容。

步骤 2:实现延迟加载的 Dart 库

接下来,在 Dart 代码中实现延迟加载的 Dart 库。实现并非立刻需要的功能。 文章剩余部分中的示例添加了一个简单的延迟 widget 作为占位。 你还可以通过修改 loadLibrary()Futures 后面的延迟加载代码的导入和保护用法,将现有代码转换为延迟代码。

  1. 创建新的 Dart 库。例如,创建一个可以在运行时下载的 `DeferredBox` widget。 这个 widget 可以是任意复杂的,本指南使用以下内容创建了一个简单的框。

// box.dart

import ‘package:flutter/widgets.dart’;

/// A simple blue 30x30 box.
class DeferredBox extends StatelessWidget {
DeferredBox() {}

@override
Widget build(BuildContext context) {
return Container(
height: 30,
width: 30,
color: Colors.blue,
);
}
}

  1. 在应用中使用 `deferred` 关键字导入新的 Dart 库,并调用 `loadLibrary()`。 下面的示例使用 `FutureBuilder` 等待 `loadLibrary` 的 `Future` 对象(在 `initState` 中创建)完成, 并将 `CircularProgressIndicator` 做为占位。 当 `Future` 完成时,会返回 `DeferredBox`。 `SomeWidget` 便可在应用程序中正常使用,在成功加载之前不会尝试访问延迟的 Dart 代码。

import ‘box.dart’ deferred as box;

// …

class SomeWidget extends StatefulWidget {
@override
_SomeWidgetState createState() => _SomeWidgetState();
}

class _SomeWidgetState extends State {
Future _libraryFuture;

@override
void initState() {
_libraryFuture = box.loadLibrary();
super.initState();
}

@override
Widget build(BuildContext context) {
return FutureBuilder(
future: _libraryFuture,
builder: (BuildContext context, AsyncSnapshot snapshot) {
if (snapshot.connectionState == ConnectionState.done) {
if (snapshot.hasError) {
return Text(‘Error: ${snapshot.error}’);
}
return box.DeferredBox();
}
return CircularProgressIndicator();
},
);
}
}
// …

loadLibrary() 函数返回一个 Future<void> 对象, 该对象会在延迟库中的代码可用时成功返回,否则返回一个错误。 延迟库中所有的符号在使用之前都应确保 loadLibrary() 已经完成。 所有导入的库都必须通过 deferred 标记,以便对其进行适当的编译以及在延迟组件中使用。 如果组件已经被加载,再次调用 loadLibrary 将快速返回(但不是同步完成)。 也可以提前调用 loadLibrary() 函数进行预加载,以帮助屏蔽加载时间。

你可以在 Flutter Gallery’s lib/deferred_widget.dart 中找到其他延迟加载组件的示例。

步骤 3:构建应用程序

使用以下 flutter 命令构建延迟组件应用:

$ flutter build appbundle

此命令会帮助你检查项目是否正确设置为构建延迟组件应用。 默认情况下,验证程序检测到任何问题都会导致构建失败,你可以通过系统建议的更改来修复这些问题。

你可以使用 --no-deferred-components 标志禁用构建延迟组件。 这个标志会让 pubspec.yaml 中定义的所有延迟组件,被视为定义在 assets 部分的普通组件。 所有 Dart 代码会被编译到一个共享库中,loadLibrary() 调用会在下一个事件循环中完成(异步时尽快完成)。 此标志也等效于移除 pubspec.yaml 中的 deferred-components:

`flutter build appbundle` 命令会尝试构建应用, 通过 `gen_snapshot` 将应用中拆分的 AOT 共享库分割为单独的 `.so` 文件。 第一次运行时,验证程序可能会在检测到问题时失败, 该工具会为如何设置项目和解决这些问题提供建议。

验证程序分为两个部分:预构建和生成快照后的验证。 这是因为在 gen_snapshot 完成并生成最后一组加载单元之前,无法执行任何引用加载单元的验证。

你可以通过 --no-validate-deferred-components 标志,来让工具尝试在不执行验证程序下构建应用。 这可能导致由意外和错误的指令而引起的故障。 此标志应当仅在不需要依赖验证程序检查的默认 Play-store-based 的自定义实现时使用。

验证程序会检测 gen_snapshot 生成的所有新增、修改或者删除的加载单元。 当前生成的加载单元记录在 <projectDirectory>/deferred_components_loading_units.yaml 文件中。 这个文件应该加入到版本管理中,以确保其他开发人员对加载单元所做的更改可被追踪。

验证程序还会检查 android 目录中的以下内容:

  • 每个延迟组件名称的键值对映射 `KaTeX parse error: Expected group as argument to '\`' at end of input: …tName}Name\`:\`{componentName}`。 每个功能模块的 `AndroidManifest.xml` 使用此字符串资源来定义 `dist:title property`。例如:
<?xml version="1.0" encoding="utf-8"?> ... boxComponent
  • 每个延迟组件都有一个 Android 动态功能模块,它包含一个 `build.gradle` 和 `src/main/AndroidManifest.xml` 文件。 验证程序只检查文件是否存在,不验证文件内容。如果文件不存在,它将生成一个默认的推荐文件。

  • 包含一个 meta-data 键值对,对加载单元与其关联的组件名称之间的映射进行编码。 嵌入程序使用此映射将 Dart 的内部加载单元 id 转换为要安装的延迟组件的名称。例如:






gen_snapshot 验证程序在预构建验证通过之前不会运行。

  1. 对于每个检查,该工具会创建或者修改需要的文件。 这些文件放在 `/build/android_deferred_components_setup_files` 目录下。 建议通过复制和覆盖项目 `android` 目录中的相同文件来应用更改。 在覆盖之前,当前的项目状态应该被提交到源代码管理中,并检查建议的改动。 该工具不会自动更改 `android` 目录。

一旦生成可用的加载单元并将其记录到 `deferred_components_loading_units.yaml` 中, 便可完善 pubspec 的 `deferred-components` 配置,将加载单元分配给延迟的组件。 在上面的案例中,生成的 `deferred_components_loading_units.yaml` 文件将包含:

loading-units:

  • id: 2
    libraries:
  • package:MyAppName/box.Dart

加载单元 id(在本例中为「2」)由 Dart 内部使用,可以忽略。 基本加载单元(id 为「1」)包含了其他加载单元中未显式列出的所有内容,在这里没有列出。

现在可以将以下内容添加到 pubspec.yaml 中:


flutter:

deferred-components:

  • name: boxComponent
    libraries:
  • package:MyAppName/box.Dart

将加载单元分配到延迟组件,把加载单元中的任何 Dart 库添加到功能模块的 libraries 部分。 请记住以下准则:

  • 一个加载单元只能包含在一个延迟组件中

  • 引用加载单元中的一个 Dart 库意味着整个加载单元都被包含在延迟组件中。

  • 所有未被分配给延迟组件的加载单元都包含在基本组件中,基本组件始终隐式存在。

  • 分配给同一延迟组件的加载单元将一起下载、安装和运行。

  • 基本组件是隐式的,不需要在 pubspec 中定义。

  1. 静态资源也可以通过在延迟组件中配置 assets 进行添加 :

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

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

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

img

img

img

img

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

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

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

最后

总之啊,家里没矿的同学们,如果你们想以后的日子过得好一些,多想想你们的业余时间怎么安排吧;

技术方面的提升肯定是重中之重,但是技术外的一些“软实力”也不能完全忽视,很多时候升职确实是因为你的技术足够强,但也与你的“软实力”密切相关

在这我也分享一份大佬自己收录整理的 Android学习PDF+架构视频+面试文档+源码笔记 ,还有高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这些都是我闲暇还会反复翻阅并给下属员工学习的精品资料。在脑图中,每个知识点专题都配有相对应的实战项目,可以有效的帮助大家掌握知识点。

总之也是在这里帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习

相信自己,没有做不到的,只有想不到的

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

升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习

[外链图片转存中…(img-EkSpPre8-1711930981742)]

[外链图片转存中…(img-sfJt0h9u-1711930981742)]

相信自己,没有做不到的,只有想不到的

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 15
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值