Web 平台支持
Fory Dart 通过生成的序列化器和平台专用运行时实现,支持 Dart VM/AOT、Flutter、浏览器以及 Flutter web 构建。这些平台上的公开 API 和注册流程相同,但 web 构建有更严格的整数精度规则,因为 Dart int 由 JavaScript number 表示。
支持的目标
Dart 运行时支持:
- Dart VM/JIT 应用。
- Dart AOT/native 应用。
- Flutter 移动端和桌面端应用。
- 编译为浏览器 JavaScript 的 Dart 应用。
- Flutter web 应用。
- 在所有受支持目标上使用生成的
@ForyStruct序列化器和手动注册的序列化器。
必须使用代码生成
Fory Dart 使用显式注册,而不是运行时反射。对于带注解的 struct,请先运行代码生成并注册生成的序列化器,然后再序列化或反序列化值:
import 'package:fory/fory.dart';
part 'account.fory.dart';
()
class Account {
Account();
String name = '';
Int64 sequence = Int64(0);
}
void main() {
final fory = Fory();
AccountFory.register(
fory,
Account,
namespace: 'example',
typeName: 'Account',
);
final bytes = fory.serialize(Account()..name = 'web');
final account = fory.deserialize<Account>(bytes);
print(account.name);
}
在构建或测试前生成配套文件:
cd dart/packages/fory
dart run build_runner build --delete-conflicting-outputs
注册调用在 VM/AOT、Flutter 和 web 上相同。手写序列化器使用 registerSerializer(...);生成的 struct 使用生成的 register 包装器。
64 位整数规则
Dart VM 的 int 值是有符号 64 位值。Dart web 的 int 值由 JavaScript number 支撑,只在 JavaScript 安全整数范围内精确:
-9007199254740991 <= value <= 9007199254740991
选择字段类型时使用以下规则:
| 逻辑值 | web 上推荐的 Dart 字段类型 | 说明 |
|---|---|---|
| JS 安全范围内的有符号 64 位值 | int | 可用于默认 int64 映射以及 @ForyField(type: Int64Type(...)) 编码。 |
| 完整的有符号 64 位范围 | Int64 | 保留 JS 安全范围之外的值。 |
| 无符号 64 位值 | Uint64 | 对无法放入有符号或 JS 安全 Dart int 的值是必需的。 |
| 8/16/32 位整数 | int + @ForyField(type: ...) | 使用显式字段元信息与对端运行时精确匹配。 |
@ForyField(type: Int64Type(...)) 控制 Dart int 字段的编码格式:
()
class SafeCounter {
SafeCounter();
(type: Int64Type(encoding: Encoding.tagged))
int count = 0; // 将 web 值保持在 JS 安全范围内
}
它不会让 Dart int 在 web 上能够存储所有 64 位值。对于完整范围的有符号值,请使用 Int64:
()
class FullRangeCounter {
FullRangeCounter();
Int64 count = Int64(0);
}
对于无符号值,请使用 Uint64:
()
class StorageExtent {
StorageExtent();
Uint64 byteOffset = Uint64(0);
}
自定义序列化器
自定义序列化器可以在 VM/AOT、Flutter 和 web 上使用相同的 Buffer、WriteContext 与 ReadContext API。对于 64 位值:
- 对完整范围的有符号 64 位值,使用
buffer.writeInt64(Int64(...))和buffer.readInt64()。 - 对完整范围的无符号 64 位值,使用
buffer.writeUint64(Uint64(...))和buffer.readUint64()。 - 仅当值本来就要作为 Dart
int使用,因此在 web 上必须保持 JS 安全时,才使用writeInt64FromInt、writeVarInt64FromInt以及匹配的AsInt读取方法。
示例:
final class OffsetSerializer extends Serializer<StorageExtent> {
const OffsetSerializer();
void write(WriteContext context, StorageExtent value) {
context.buffer.writeUint64(value.byteOffset);
}
StorageExtent read(ReadContext context) {
return StorageExtent()..byteOffset = context.buffer.readUint64();
}
}
集合和类型化数组
List、Set、Map、Uint8List、数值类型化数组、Int64List 和 Uint64List 在 web 上都受支持。Int64List 和 Uint64List 的实现会保留 64 位值,不依赖 JavaScript 整数精度。当 schema 是 array<int64> 或 array<uint64> 时,请使用 Fory 包装 list 类型。
测试浏览器构建
修改必须在 web 上工作的代码时,请同时在 VM 和 Chrome 中运行包测试:
cd dart/packages/fory
dart run build_runner build --delete-conflicting-outputs
dart test
dart test -p chrome
如果 Chrome 测试因生成文件过期或缺少 part 文件而失败,请重新运行 build_runner,然后从 dart/packages/fory 目录重试测试命令。
常见 Web 失败
Dart int value ... is outside the JS-safe signed int64 range
序列化器正尝试在 web 上把 Dart int 写成有符号 64 位值,但该值超出了 JavaScript number 可以精确表示的范围。请将字段类型改为 Int64,或把值保持在 JS 安全范围内。
Int64 value ... is not a JS-safe int
反序列化器读取了完整范围的 Int64,但目标字段或自定义序列化器要求 Dart int。请将字段类型改为 Int64,或使用 readInt64() 解码,而不是使用 AsInt 辅助方法。
Uint64 value ... is not a JS-safe int
代码正在 web 上把 Uint64 转换为 Dart int。除非应用已经验证该值位于 JS 安全的非负范围内,否则请保持为 Uint64。