Flutter 3.0已經與2022年5月12日發布,隨之發布的還有 Dart 2.17版本,在感嘆谷歌的版本升級快節奏之余,也來看看 Dart 2.17版本有哪些新特性吧。
概述
本次發布的版本主要是為了提高生產力和提高跨平臺兼容性。同時提供了新的語言特性,具體如下:
- 支撐成員枚舉
- 改善了父類參數轉發的方式
- 命名參數更加靈活
同時更新了 Dart 語言編碼規范檢查工具(package:lints),以遵循最佳的 Dart 編程實踐。 此外,還更新了核心庫的 API 文檔,提供了更豐富的示例代碼。為了提高跨平臺的可移植性,在 Flutter 插件中提供了使用 dart:ffi(用于與原生 C 語言代碼交互)新的模板代碼,以便支持 RISC-V 指令集處理器,以及支持對 macOS和 Windows 應用程序進行簽名。
枚舉特性增強,支持成員屬性
這個特性其實在 Java 語言中已經支持,其實就是支持在枚舉中支持定義成員,包括變量和方法。這樣的好處是可以在構建枚舉的時候指定對應的值和支持調用枚舉的方法。這樣的枚舉更像是一個類了。舉個例子,我們會對水會定義冰水(frozen,0攝氏度)、溫水(40攝氏度)和開水(100攝氏度),以前要獲取一個枚舉對應的值,需要使用 extension
進行擴展,示例代碼如下:
enum Water {
frozen,
lukewarm,
boiling;
}
extension Members on Water {
int waterToTemp(Water water) {
switch (water) {
case Water.frozen:
return 0;
case Water.lukewarm:
return 40;
case Water.boiling:
return 100;
}
}
String convertToString() => 'The $name water is ${waterToTemp(this)} ℃.';
}
void main() {
print(Water.frozen.convertToString());
}
當枚舉對應特定值的時候,這樣就太麻煩了!Dart 2.17版本后就簡單多了。
enum Water {
frozen(0),
lukewarm(40),
boiling(100);
final int temperature;
const Water(this.temperature);
@override
String toString() => 'The $name water is $temperature ℃.'
}
void main() {
print(Water.frozen);
}
簡化父類構造器
當我們在 Dart 中使用繼承的時候,通常會需要調用父類的構造器進行初始化。之前我們通常需要先列出類自身的構造方法的參數,然后再把這些參數傳給父類,就系那個下面這樣:
class OutlineButton extends ButtonStyleButton {
const OutlineButton({
Key? key,
required VoidCallback? onPressed,
VoidCallback? onLongPress,
ValueChanged<bool>? onHover,
ValueChanged<bool>? onFocusChange,
ButtonStyle? style,
FocusNode? focusNode,
bool autofocus = false,
Clip clipBehavior = Clip.none,
required Widget child,
}) : super(
key: key
onPressed:onPressed,
onLongPress:onLongPress,
onHover:onHover,
onFocusChange:onFocusChange,
...
說實話寫起來確實很繁瑣,簡直就是廢話一般!升級到 Dart 2.17后,使用 super 關鍵字直接引用即可,就像下面這樣,而且不需要在調用父類方法了,清爽太多了!
class OutlineButton extends ButtonStyleButton {
const OutlineButton({
super.key,
super.onPressed,
super.onLongPress,
super.onHover,
super.onFocusChange,
super.style,
super.focusNode,
super.autofocus = false,
super.clipBehavior = Clip.none,
required Widget super.child,
});
...
...
}
隨處可見的命名參數優化
之前版本中,命名參數需要在匿名參數后面,這種寫法需要我們知道前面的參數,必須嚴格按照次序去寫代碼,一方面是寫代碼的時候需要知道參數次序,另一方面是有時候會導致代碼可讀性下降。現在的話可以將命名參數提前到任意位置(但是匿名參數還是需要按次序),以 List.generate 方法為例,我們對比一下:
//之前的寫法:growable 必須放在后面
final factorials = List<int>.generate(
10,
(int i) {
if (i == 0) {
return 1;
} else {
var result = 1;
for (var r = 2; r <= i; ++r) {
result *= r;
}
return result;
}
},
growable: true,
);
// Dart 2.17后,growabe位置可以提前
final factorials = List<int>.generate(
10,
growable: true,
(int i) {
if (i == 0) {
return 1;
} else {
var result = 1;
for (var r = 2; r <= i; ++r) {
result *= r;
}
return result;
}
},
);
生產力工具
一個是代碼檢查工具的升級,Dart 代碼檢查工具增加了10條新的規范,還為Flutter 開發增加了2條特定的規范。可以通過下面的命令進行升級:
# Dart 開發
dart pub upgrade —-major-versions lints
# Flutter 開發
flutter pub upgrade —-major-versions flutter_lints
對于加密的 Socket (SecureSockets)調試增加了加密數據查看的支持,具體來說就是通過指定一個 keyLog 文件,當和服務端交換 TLS 密鑰的時候,以 NSS Key Log Format
(一種Mozila 定義的格式)的一行文本會添加到這個文件中,這使得一些網絡流量分析工具(例如 WireShark)可以將 socket 發送的內容進行解密,從而方便調試。 具體可以查看 SecureSocket.connect()
的 API文檔。
此外, Dart 開發團隊聽取了開發者更喜歡閱讀示例的方式理解 API 的建議,在核心庫最常用的200個 API 中增加了示例代碼,使得閱讀和理解 API 更容易。
對于棄用的代碼特性(共231條),可以通過 dart fix
命令直接替換。
Dart FFI
Dart FFI 是用于 Dart 代碼和C語言/原生代碼交互的機制。當需要使用原生 API 開發 Flutter 插件時,使用 Dart FFI會更方便。Dart 2.17為了簡化這樣的開發工作,在 Flutter 中提供了模板代碼,具體可以到官網了解如何使用:開發軟件包和插件指南。
總結
這就是Dart 2.17的新特性介紹,可以看到目前 Dart 團隊主要集中在跨平臺和生產力提升上。說明整個 Dart 語言已經到了穩定的階段,接下來的更新迭代將會讓我們的開發效率更高。