【问题标题】:Dart null safety and parsing JSON to models with json_serializableDart null 安全性和使用 json_serializable 将 JSON 解析为模型
【发布时间】: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 来检查成员是否为空。我认为避免这种情况是健全的空安全性背后的想法之一

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

【问题讨论】:

    标签: dart json-serializable


    【解决方案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)
    }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2021-08-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-08-04
      • 2020-02-14
      • 2021-10-01
      相关资源
      最近更新 更多