当前位置: 首页 > news >正文

flutter json解析增强

依赖:xxf_json

反序列化兼容特征一览表

类型\是否兼容

int

double

num

string

bool

int

yes

yes

yes

yes

yes

double

yes

yes

yes

yes

yes

num

yes

yes

yes

yes

yes

string

yes

yes

yes

yes

yes

bool

yes

yes

yes

yes

yes

专业词语

.g.dart : 是json_annotation生成的中间解析文件

DTO : 网络传输模型,这里泛指json解析模型

中间件

对json提供如下基础中间件,两种兼容模式,一种给全日志

  1. JsonConverter 兼容基本类型(int,num,double,string,bool),异常情况解析成对应类型的默认值,达到同级别js,oc等语言层兼容

  2. nullable_converter 兼容基本类型(int,num,double,string,bool),异常情况解析成null,需要DTO声明字段为可空类型

  3. strict_converter 先兼容解析,解析不了再报错,解决json解析报错,不提示具体内容,导致一个一个去比较DTO里声明的字段,或者打印stack才可以排查具体字段

至于要使用多少类型兼容和什么策略,请自己选,上图只是模版

用法

  1. DTO层增加注解 通过 @JsonSerializable 注解参数converters 注入!

其中 primitiveConvertors :基本类型解析器,安全处理,如遇到失败会转换成默认值 primitiveNullableConvertors:基本类型解析器,安全处理,如遇到失败变成null,适合声明可空类型的字段

primitiveStrictConvertors:基本类型解析器,不安全处理,用于提示内容,比原错误始信息增加 内容本身到日志里面,解决原始报错,不提示具体内容,不好分析DTO的那个字段

用法代码示例:

@JsonSerializable(converters: [...primitiveNullableConvertors, RRuleJsonAdapter()])
class Event{
}

  1. 字段级别增加注解

通过 @JsonKey fromJson 注解增加参数

用法代码示例:

@JsonSerializable()
class Event  {//类型不对,就解析成""双引号字符串@JsonKey(fromJson: StringDecoder.decodeOrEmpty)String? eventId; //设备内日程id
}

提供更自由的控制,每种类型都提供三种策略

bool_decoder.dart

double_decoder.dart

int_decoder.dart

num_decoder.dart

string_decoder.dart

class StringDecoder {///策略1 异常情况解析为空static String? decodeOrNull(dynamic json) {if (json == null) return null;return json.toString();}///策略2 异常情况解析默认值,比@JsonKey(defaultValue: "")更健壮static String decodeOrEmpty(dynamic value) {return decodeOrNull(value) ?? "";}///策略3 尝试解析异常情况报错并给出错误值static String decodeOrException(dynamic value) {return decodeOrNull(value) ??(throw BaseDecoder.createParseError("String", value));}
}

框架的中间件优先级是 @JsonKey(fromJson)>@JsonSerializable(converters)

推荐倾向

1. DTO里字段声明成可空字段

给DTO增加可空兼容转换器@JsonSerializable( converters: primitiveNullableConvertors),

具体字段的应用由业务层来处理兼容, 这样不至于整个页面出问题,个别字段的问题,交由service层/repo层来校验参数,

eg.如在版本迭代中 枚举的类型可能增加,但是之前版本代码是没有的,那么非空类型就出问题了

实在要坚持后端一定不会变,其他页面传进来的参数也不会变, 那么就选择primitiveStrictConvertors或者decodeOrException 会尝试解析之后再报错出来!

规范

结合convertor和 fromJson,toJson Api 注入到DTO身上,不应该去手写解析 手写/或者手改.g.dart 合并代码和以及兼容性都有些问题

枚举的兼容

枚举默认按名字,如按其他值解析,有如下三种方式 1. 在枚举值上添加@JsonValue(value)

  1. 自定义jsonDecoder 这样枚举在其他DTO声明的地方都可以快速适配,

  2. 在其他DTO 声明枚举的字段上增加注解 @JsonKey(fromJson: StatusConverter.fromJson, toJson: StatusConverter.toJson)

不要在其他DTO解析的地方(其他DTO有声明这个枚举类型字段) 来手写if判断!


/// 日历账号类型
enum CalDavTypeEnum {google("google"),iCloud("iCloud"),calDAV("CalDAV");final String value;const CalDavTypeEnum(this.value);
}/// 枚举值和 JSON 数据的映射关系
Map<String, CalDavTypeEnum> _stringToEnum =CalDavTypeEnum.values.associateBy((e) => e.value);
Map<CalDavTypeEnum, String> _enumToString =CalDavTypeEnum.values.associate((e) => MapEntry(e, e.value));class CalDavTypeEnumDecoder {static CalDavTypeEnum decode(dynamic json) {return _stringToEnum["$json"] ??(throw ArgumentError('Unknown enum value: $json'));}static CalDavTypeEnum? decodeOrNull(dynamic json) {return _stringToEnum["$json"];}
}class CalDavTypeEnumEncoder {static String encode(CalDavTypeEnum? myEnum) {return _enumToString[myEnum] ??(throw ArgumentError('Unknown enum: $myEnum'));}static String? encodeOrNull(CalDavTypeEnum? myEnum) {return _enumToString[myEnum];}
}///解析转换器
class CalDavTypeEnumJsonConverterextends JsonConverter<CalDavTypeEnum, dynamic> {const CalDavTypeEnumJsonConverter();@overrideCalDavTypeEnum fromJson(json) {return CalDavTypeEnumDecoder.decode(json);}@overridetoJson(CalDavTypeEnum object) {return CalDavTypeEnumEncoder.encode(object);}
}///解析转换器 可空
class CalDavTypeEnumNullableJsonConverterextends JsonConverter<CalDavTypeEnum?, dynamic> {const CalDavTypeEnumNullableJsonConverter();@overrideCalDavTypeEnum? fromJson(json) {return CalDavTypeEnumDecoder.decodeOrNull(json);}@overridetoJson(CalDavTypeEnum? object) {return CalDavTypeEnumEncoder.encodeOrNull(object);}
}

json_annotation使用指引
 

  1. 模型增加 @JsonSerializable注解

  2. 在模型声明文件头部增加 part 'xxx.g.dart'; 其中xxx 一般是模型名字,当然也可以是其他名字

  3. 字段如有必要增加@JsonKey

  4. 在命令行 cd 到模型对应的目录

  5. 在命令行执行 flutter pub run build_runner build

  6. 那么就能看到生成 xxx.g.dart文件的生成,将这个文件添加到git

  7. 然后再模型里面声明方法引用

part 'account_xxx.g.dart';@JsonSerializable()
class AccountInfo {int? id;//声明反序列化方法factory AccountInfo.fromJson(Map<String, dynamic> json) =>_$AccountInfoFromJson(json);///声明序列化方法Map<String, dynamic> toJson() => _$AccountInfoToJson(this);
}

相关文章:

  • Android 9.0系统源码定制:实现开机启动特定App的全面指南
  • 《分布式软总线:不同频段Wi-Fi环境下设备发现兼容性难题》
  • leetcode面试经典算法题——2
  • 微店商品详情API接口:功能解析与数据应用实践
  • LLM-as-Judge真的更偏好AI输出?
  • 鸿蒙应用元服务开发-Account Kit配置登录权限
  • Prometheus架构组件
  • 国内开源医疗模型研究报告
  • 自动化测试工具playwright中文文档-------14.Chrome 插件
  • 如何在NS3中搭建窄带干扰和扫频干扰场景?
  • 844. 比较含退格的字符串
  • 安装SQLServer管理工具
  • 日语学习-日语知识点小记-构建基础-JLPT-N4阶段(4): 可能形(かのうけい)
  • Coze平台技术解析:零代码AI开发与智能体应用实践
  • 跑得快的标准详细规则·棒球1号位
  • OSPF协议基础
  • 《分布式软总线:网络抖动下的数据传输“定海神针”》
  • 【四川省第三届青少年C++算法设计大赛 (小低组) 第 一试】
  • 【Java学习】全局异常处理器的使用流程
  • opencv二值化实验
  • 新质观察|“模速空间”如何成为“模范空间”
  • 长三角9座“万亿城市”一季报出炉:多地机器人产量大增
  • 甘肃临夏州政协委员马全成涉嫌诈骗罪,被撤销政协委员资格
  • 世锦赛决赛今夜打响,斯诺克运动需要赵心童创造历史
  • 中国驻日本大使吴江浩就日本民用飞机侵闯我钓鱼岛领空向日方提出严正交涉
  • 遍体鳞伤就是击不倒,这是国米老男孩最后的倔强