Dart null 安全性和使用 json_serializable 将 JSON 解析为模型

Posted

技术标签:

【中文标题】Dart null 安全性和使用 json_serializable 将 JSON 解析为模型【英文标题】:Dart null safety and parsing JSON to models with json_serializable 【发布时间】:2021-11-12 16:18:19 【问题描述】:

在使用各种工具将 JSON 反序列化为 Dart 并且对 null 安全性 感到非常沮丧之后,我有一个非常普遍的问题。生成.fromJson.toJson 消息的json_serializable 包应该关心NULL 值。但我根本无法做到这一点!所以一百万美元的问题是:

我是否必须将模型的所有属性声明为 NULLABLE(使用 ? 例如 String? myString 如果有任何机会该属性可能是 在返回的 JSON 中 >missing 或 NULL ??

当我不将此类成员声明为 NULLABLE 时,我总是会收到此错误:

Unhandled Exception: type 'Null' is not a subtype of type 'List<dynamic>' in type cast

这些错误总是发生在生成的FromJson 方法中。这是导致问题的示例 JSON:

  "responseStatus": 
    "errorCode": "500",
    "message": "This is a test message",
    "errors": [
      
        "errorCode": "300",
        "fieldName": "a field name",
        "message": "a message",
        "meta": 
          "a key": "a value"
        
      
    ],
    "meta": 
      "some field": "some value"
    
  

errors 数组包含 error objects 数组,它本身可能包含也可能不包含 key value pairs (meta) 数组。如果有 NO 错误,则errors 数组未提交,如下所示:

"responseStatus": 
    "errorCode": "500",
    "message": "This is a test message",
    "meta": 
      "some field": "some value"
    

在这种情况下,生成的 DART 代码会崩溃,如下所示:

ResponseStatus _$ResponseStatusFromJson(Map<String, dynamic> json) =>
    ResponseStatus(
      errorCode: json['errorCode'] as String? ?? '',
      message: json['message'] as String? ?? '',
      stackTrace: json['stackTrace'] as String? ?? '',
    )
      ..errors = (json['errors'] as List<dynamic>)  // <<==== This is crashing if errors is not in the returned JSON
          .map((e) => ResponseError.fromJson(e as Map<String, dynamic>))
          .toList()
      ..meta = Map<String, String>.from(json['meta'] as Map);

我创建的模型类如下:

@JsonSerializable(explicitToJson: true)
class ResponseStatus extends Equatable 
  late final String errorCode;
  late final String message;
  late final String stackTrace;
  late final List<ResponseError> errors;
  late final Map<String, String> meta;


  ResponseStatus(this.errorCode = '', this.message = '', this.stackTrace = '') 
    this.errors = [];
    this.meta = Map();
  

  factory ResponseStatus.fromJson(Map<String, dynamic> json) =>
      _$ResponseStatusFromJson(json);

  Map<String, dynamic> toJson() => _$ResponseStatusToJson(this);

  @override
  List<Object?> get props => [errorCode, message, stackTrace, errors, meta];


我创建了一个默认构造函数,它用一个值初始化所有成员(不可为空)。

如果我必须将所有模型成员声明为 nullable,这将无法使用,因为在我的颤振小部件中,我必须编写数千个 if statements 来检查成员是否为空。我认为避免这种情况是健全的空安全性背后的想法之一

我很想知道这是否是一个错误,或者我是如何让这个东西工作的!

【问题讨论】:

【参考方案1】:

这是预期的行为。如果未给出数据,则必须发生崩溃,因为不可为空的变量永远不可能有null 值。

但是,您可以使用自定义反序列化函数来处理这种情况:

@JsonKey(fromJson: optionalErrorListFromJson)
late final List<ResponseError> errors;
List<ResponseError> optionalErrorListFromJson(List<dynamic>? errors) 
  if (errors == null) return const [];
  return errors
      .map((e) => ResponseError.fromJson(e as Map<String, dynamic>))
      .toList(growable: false)

【讨论】:

以上是关于Dart null 安全性和使用 json_serializable 将 JSON 解析为模型的主要内容,如果未能解决你的问题,请参考以下文章

▩Dart-空安全(Null Safety)

使用 if 检查进行 Dart 空安全 [重复]

Dart - 如果值不是分数,则 int.toDouble() 仍将 runTimeType 设为 int

Dart null 安全性不适用于类字段

库 'package:flutter/material.dart' 是遗留的,不应导入到 null 安全库中

是否可以为依赖的 Dart 包禁用空安全性?