Dart 2.17 发布 | 语言特性、生产力工具、更深入和更广泛的平台集成能力

官方发布说明:medium.com/dartlang/da…

Google I/O 上,我们发布了新的 Dart 2.17 版本 SDK。此版本构建于我们领先的生产力和平台可移植性的核心之上。新版本提供了新的语言特性:枚举支持成员变量、改进的超类参数继承,以及更为灵活的命名参数。我们为 package:lints 提升了主版本,它是根据我们的最佳实践检查 Dart 代码的规则集。同时也更新了拥有丰富代码示例的核心库的 API 文档。为了改进平台集成的能力,我们提供了在 Flutter 插件中使用 dart:ffi 的新模板、对 RISC-V 处理器的实验性支持以及对 macOS 和 Windows 可执行文件的签名支持。

image-20220512063028216

生产力提升的语言新特性

我们一直在持续地改进 Dart 语言,通过添加新功能和改进现有的功能,以提高你的工作效率。Dart 2.17 增加了对枚举的成员变量的支持,优化了在构造函数中使用命名参数的方式,并且开始使用继承超类的参数以减少冗长和重复的代码。

增强的支持成员变量的枚举

枚举非常适合表示一组离散的状态。例如,我们可以将水描述为 enum Water { frozen, lukewarm, boiling }。但如果我们想在 enum 上添加一些方法,例如将每个状态转换为温度,并支持将 enum 转换为 String,该怎么办?或许我们可以使用扩展方法来添加一个 waterToTemp() 方法,但我们必须时刻注意它与 enum 的同步。对于 String 我们希望重载 toString() 方法,但它不受支持。

在 Dart 2.17 中,枚举类的成员变量已支持。这意味着我们可以添加保存状态的字段、设置状态的构造函数、具有功能的方法,甚至重载现有的方法。社区中许多开发者一直有这样的需求,这是我们在语言仓库的问题追踪中 投票率第三的问题

对于我们的 Water 示例,我们可以添加一个保存温度的 int 字段,并添加接收 int 的默认构造函数:

enum Water 
  const Water(this.tempInFahrenheit);
​
  final int tempInFahrenheit;
}
复制代码

为了确保在创建枚举时构造被正常调用,我们需要为每一个枚举值进行附以显式的调用:

enum Water {
  frozen(32),
  lukewarm(100),
  boiling(212);
}
复制代码

想要支持从枚举转换为 String,我们可以很简单地重载 toString 方法,因为 enums 也继承自 Object

@override
String toString() => "The $name water is $tempInFahrenheit F.";
复制代码

如此一来,你就有了一个可以轻松实例化完整功能的枚举类,并且可以在任意位置调用方法:

void main() {
  print(Water.frozen); // 打印 “The frozen water is 32 F.”
}
复制代码

这两种方法的完整示例如下所示。有了这些改动,新版本的代码更易于阅读和维护。

image-20220512081539602

超类的初始化构造

当你的代码存在类型继承关系式,一个常见的范式是将一些构造函数参数传递给超类的构造函数。为此子类需要 1) 在其构造函数中列出每个参数 2) 使用这些参数调用超类的构造函数。这导致了大量的代码重复,使代码难以阅读和维护。

而几位 Dart 社区成员帮助 Dart 实现了这项语言目标。半年前,GitHub 用户 @roy-sianez 提交了一个 语言问题。他的建议类似于 GitHub 用户 @apps-transround 先前的 建议:也许我们可以通过引入一个新的方式来表示在超类中指定了一个参数,来解决这个问题。我们认为这是一个好主意,因此已将其实现并添加到了 Dart 2.17 版本中。从以下示例中可以看出,这与 Flutter widget 的代码有很强的相关性。实际上当我们将这项特性应用到 Flutter 框架时,我们看到框架总共减少了 近两千行代码

img

可在任意参数位置使用命名参数

最后,我们改进了调用方法的命名参数的方式。在此次更新之前,命名参数的调用必须出现在普通参数列表的后面。当你想要提升代码可读性从而希望命名参数写在靠前的位置,但无法这么做时,会觉得非常烦人。例如下方 List<T>.generate 构造函数的调用。此次更新之前 growable 参数必须放在最后,这会导致这个参数很容易被可能有很多内容的构造参数所影响而错过。现在你可以根据自己的喜好对它们进行排序,你可以先使用命名参数,最后使用生成器参数。

image-20220512081451673

更多有关这三项改进的示例,参阅我们更新的 枚举超类的初始化构造命名参数 示例。

生产力工具

回到生产力的主题,我们围绕生产力对核心工具进行了一些改进。

在 Dart 2.14 中,我们引入了 package:lints,它与 Dart 分析器一起工作,防止你编写错误的代码,并使用更规范的规则审查你的 Dart 代码。自那之后分析器中又新增了许多代码提示规则,我们对其进行了仔细分类,并从中选择了 10 条新的用于所有 Dart 代码的代码提示规则 ,以及 2 条新的专门用于 Flutter 代码的代码提示规则。它们包括确保你导入的 package 有正确地在你的 pubspec 文件中声明、防止滥用对类型参数的空检查以及确保子属性的格式一致的代码提示规则。你可以简单地使用命令升级到新的 lints package:

  • 对于 Dart package: dart pub upgrade —-major-versions lints
  • 对于 Flutter package: flutter pub upgrade —-major-versions flutter_lints

SecureSockets 通常用于启用使用 TLS 和 SSL 保护的 TCP 套接字连接。在 Dart 2.17 之前,因为没有办法检查安全数据流量,在开发过程中调试这些连接十分棘手。现在我们添加了对指定 keyLog 文件的支持。指定后,当与服务器交换新的 TLS 密钥时,NSS 密钥日志格式 中的一行文本将附加到文件中。这将使网络流量分析工具(例如Wireshark)能够解密通过套接字发送的内容。有关详细信息,请参阅SecureSocket.connect()API 文档

dart doc 生成的 API 文档是大多数 Dart 开发人员学习新 API 的重要内容之一。虽然我们的 核心库 API 长期以来都有丰富的文本描述,但许多开发人员告诉我们,他们更喜欢通过阅读 API 的示例代码来学习 API。在 Dart 2.17 中,我们检查了所有主要的核心库,将示例代码添加到浏览次数最多的前 200 个页面中,它们现在拥有非常全面的示例代码。例如,将 Dart 2.16 中 dart:convert 的文档Dart 2.17 的更新页面进行比较。希望这会使文档更易于阅读和查阅。

生产力的提高不仅源于我们向平台添加的新功能,还同时来自于我们清理的堆栈,并且删除了不再使用的功能。这有助于保持我们的代码较小,对于新的开发人员来说尤其重要。为此,我们从 dart:io 库中删除了 231 行已弃用的代码。如果你仍在使用这些已弃用的 API,你可以使用 dart fix 进行修复和替换。我们还在继续努力删除 已弃用的 Dart CLI 工具,本次更新删除了dartdoc 工具(使用dart doc 代替)和 pub 工具(使用 dart pubflutter pub 代替)。

扩大我们的平台集成和支持

第二个核心主题是平台集成和支持。Dart 是一种真正的多平台语言。虽然我们已经支持 大量的平台,但我们仍在不断开发以确保你可以与每个受支持的平台深度集成,同时支持新兴的平台。

我们 与 C/native 代码互操作 的核心机制 Dart FFI 是一种将 Dart 代码与现有原生平台代码集成的流行方式。在 Flutter 上,FFI 是构建使用宿主平台原生 API(例如 Windows win32 API)的 插件 的好方法。在 Dart 2.17 和 Flutter 3 中,我们向 flutter 工具添加了 FFI 的模板,现在你可以轻松地创建 FFI 插件,这些插件具有通过 dart:ffi 调用原生代码支持的 Dart API。详细信息请参阅 开发包和插件 页面。

FFI 现在支持特定于 ABI 的类型,可以在具有特定 ABI(应用程序二进制接口) 类型的平台上使用 FFI。例如,现在你可以使用 Long(C 语言中的 long) 正确表示具有特定于 ABI 大小的长整数,由于 CPU 架构的区别,结果可能是 32 位或 64 位。有关支持类型的完整列表,请参阅 AbiSpecificInteger API 页面 中的「Implementers」列表。

在使用 Dart FFI 与原生平台深度集成时,有时需要将 Dart 分配的内存或其他资源(端口、文件等)的清理行为与原生代码对齐。该问题长期以来都很难解决,因为 Dart 是一种会自动处理垃圾回收清理行为的语言。在 Dart 2.17 中,我们通过引入 Finalizer 的概念解决了这个问题,它包括一个 Finalizable 标记接口,用于「标记」不应过早终结或丢弃的对象,以及一个 NativeFinalizer 可以附加到 Dart 对象上,当对象即将被垃圾回收时提供回调运行。Finalizer 让原生代码和 Dart 代码中同时运行清理。更多详细信息请参阅 NativeFinalizer API 文档 中的描述和示例,或 WeakReferences 以及 Finalizer 在 Dart 代码中的类似支持。

将 Dart 编译为本机代码的支持,也是使 Flutter 应用具有出色的启动性能和快速渲染的核心。除此之外,你还可以使用 dart compile 编译 Dart 文件为可执行文件。这些可执行文件可以在任何机器上独立运行,无需安装 Dart SDK。Dart 2.17 中的另一个新功能是支持对 可执行文件进行签名,生成的产物可以在经常需要签名的 Windows 和 macOS 上进行部署。

我们还保持在新兴的平台前沿,继续扩大我们所支持的平台集。RISC-V 是用于处理器的全新指令集。RISC-V International 是一家全球性的非盈利组织,拥有 RISC-V 规范,使得指令集自由开放。这仍然是一个新兴的平台,但我们对其潜力感到兴奋,因此我们的 2.17.0–266.1.beta Linux 版本包含了对它的实验性支持。我们希望能收获你的反馈,你可以 提出问题分享 你的体验!

开始使用 Dart 2.17!

我们希望今天的 Dart 2.17 版本能激发你的兴趣,提高你的工作效率,并为你的应用程序提供更多平台集成。你可以直接下载 Dart 2.17 版本开始使用,或者使用 Flutter 3 版本 SDK 中的嵌入 Dart SDK。

我们还将继续在 Google I/O 上提供 更多的的新内容

猜你喜欢

转载自juejin.im/post/7096635408339075080