然后我们在 main() 中去调用 decodeCountry() 运行,报错了…
Unhandled Exception: type ‘List’ is not a subtype of type ‘List’
…
错误日志说 List<dynamic>
不是 List<String>
的子类型,也就是我们在country的实体类中直接给 cities 属性赋值为 cities: json['cities']
,我们先来看看 json['cities']
是什么类型:
factory Country.fromJson(Map<String, dynamic> json) {
print(‘json[“cities”] type is ${json[‘cities’].runtimeType}’);
return Country(name: json[‘name’], cities: json[‘cities’]);
}
输出如下:
flutter: json[“cities”] type is List
这个时候我们需要将 Country.fromJson(...)
方法作如下更改:
factory Country.fromJson(Map<String, dynamic> json) {
print(‘json[“cities”] type is ${json[‘cities’].runtimeType}’);
var originList = json[‘cities’];
List cityList = new List.from(originList);
return Country(name: json[‘name’], cities: cityList);
}
上述代码中,我们创建了一个 List<String>
类型的数组,然后将 List<dynamic>
数组中的元素都添加到了 List<String>
中。输出如下:
flutter: json[“cities”] type is List
flutter: country name is China
对象嵌套
定义一个 shape.json ,格式如下:
{
“name”: “rectangle”,
“property”: {
“width”: 5.0,
“height”: 10.0
}
}
实体如下:
class Shape {
String name;
Property property;
Shape({this.name, this.property});
factory Shape.fromJson(Map<String, dynamic> json) {
return Shape(name: json[‘name’], property: json[‘property’]);
}
}
class Property {
double width;
double height;
Property({this.width, this.height});
factory Property.fromJson(Map<String, dynamic> json) {
return Property(width: json[‘width’], height: json[‘height’]);
}
}
Service 类如下:
import ‘dart:async’;
import ‘dart:convert’;
import ‘package:flutter/services.dart’;
import ‘…/models/shape.dart’;
Future _loadShapeJson() async {
return await rootBundle.loadString(‘assets/shape.json’);
}
Future decodeShape() async {
String shapeJson = await _loadShapeJson();
Map<String, dynamic> jsonMap = json.decode(shapeJson);
Shape shape = Shape.fromJson(jsonMap);
print(‘shape name is ${shape.name}’);
return shape;
}
运行之后,会报如下错误:
Unhandled Exception: type ‘_InternalLinkedHashMap<String, dynamic>’ is not a subtype of type ‘Property’
也就是说 property: json['property']
这里赋值的类型是 _InternalLinkedHashMap<String, dynamic>
而不是 Property
,json['property']
的值是这样的 {width: 5.0, height: 10.0}
,它是一个 Map ,并不是一个 Property
对象,我们需要先将这个 Map 转化为对象,然后在赋值:
factory Shape.fromJson(Map<String, dynamic> json) {
print(‘json[“property”] is ${json[‘property’]}’);
Property property = Property.fromJson(json[‘property’]); // new line
return Shape(name: json[‘name’], property: property);
}
输出:
shape name is rectangle
复杂的对象数组嵌套
{
“id”: “0302”,
“class_name”: “三年二班”,
“students”: [
{
“name”: “叶湘伦”,
“sex”: “男”
},
{
“name”: “路小雨”,
“sex”: “女”
}
]
}
实体:
class ClassInfo {
String id;
String name;
List studentList;
ClassInfo({this.id, this.name, this.studentList});
factory ClassInfo.fromJson(Map<String, dynamic> json) {
return ClassInfo(
id: json[‘id’],
name: json[‘class_name’],
studentList: json[‘students’]);
}
}
class Student {
String name;
String sex;
Student({this.name, this.sex});
factory Student.fromJson(Map<String, dynamic> json) {
return Student(name: json[‘name’], sex: json[‘sex’]);
}
}
service:
import ‘dart:async’;
import ‘dart:convert’;
import ‘package:flutter/services.dart’;
import ‘…/models/class_info.dart’;
Future _loadClassInfoJson() async {
return await rootBundle.loadString(‘assets/class_info.json’);
}
Future decodeClassInfo() async {
String classInfoJson = await _loadClassInfoJson();
Map<String, dynamic> jsonMap = json.decode(classInfoJson);
ClassInfo classInfo = ClassInfo.fromJson(jsonMap);
classInfo.studentList
.forEach((student) => print(‘student name is ${student.name}’));
return classInfo;
}
上述代码在运行后还是会报错:
Unhandled Exception: type ‘List’ is not a subtype of type ‘List’
同样,还是在 studentList: json['students']
出问题了,我们把 json['students']
的输出来看看:
[{name: 叶湘伦, sex: 男}, {name: 路小雨, sex: 女}]
上述结果的类型为 List<dynamic>
。现在我们需要将 List<dynamic>
转换到一个 List<Student>
类型的数组中,这里需要用到一个操作符 map
,map
操作符的作用就是将某种类型转换为另一种类型。如下:
factory ClassInfo.fromJson(Map<String, dynamic> json) {
final originList = json[‘students’] as List;
List studentList =
originList.map((value) => Student.fromJson(value)).toList();
return ClassInfo(
id: json[‘id’], name: json[‘class_name’], studentList: studentList);
}
输出:
flutter: student name is 叶湘伦
flutter: student name is 路小雨
单纯的数组
member.json
[
{
“id”: 1,
“name”: “Jack”
},
{
“id”: 2,
“name”: “Rose”
},
{
“id”: 3,
“name”: “Karl”
}
]
实体:
class MemberList {
List memberList;
MemberList({this.memberList});
factory MemberList.fromJson(List listJson) {
List memberList =
listJson.map((value) => Member.fromJson(value)).toList();
return MemberList(memberList: memberList);
}
}
class Member {
int id;
String name;
Member({this.id, this.name});
factory Member.fromJson(Map<String, dynamic> json) {
return Member(id: json[‘id’], name: json[‘name’]);
}
}
因为 member.json 是一个单纯的数组,所以上述代码中我们创建了一个 MemberList 类来将这个 Member 数组包含起来。
注意下上述代码中 MemberList.fromJson(...)
中的写法。
service:
import ‘dart:async’;
import ‘package:flutter/services.dart’;
import ‘dart:convert’;
import ‘…/models/member.dart’;
Future _loadMemberJson() async {
return await rootBundle.loadString(‘assets/member.json’);
}
Future decodeMemberList() async {
String memberListJson = await _loadMemberJson();
List list = json.decode(memberListJson);
MemberList memberList = MemberList.fromJson(list);
memberList.memberList
.forEach((member) => print(‘member name is ${member.name}’));
return memberList;
}
输出:
flutter: member name is Jack
flutter: member name is Rose
flutter: member name is Karl
复杂的 Json 解析
之前的文章 Flutter 中 ListView 的使用 中用到了 豆瓣API ,这个 API 中返回的数据包含了当前热播的电影,大家尝试着按需解析一下吧 !!!
如有错误,还请指出。谢谢!!!
源码地址
参考链接:
medium.com/flutter-com…
github.com/flutter/flu…
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
img-C5mtgJVJ-1715514475115)]
[外链图片转存中…(img-kFYlUoxA-1715514475116)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!