KakaJSON手册的第2篇文章中提过:由于JSON格式能表达的数据类型是比较有限的,所以服务器返回的JSON数据有时无法自动转换成客户端想要的数据类型

  • 比如客户端想要的是Date类型,服务器返回的可能是字符串"2018-08-08 08:08:08.888"或者"2018/08/08 08:08:08.888"
  • 像上述情况,KakaJSON内部是无法自动转换的,但提供了值过滤机制,允许开发者对JSON值进行自定义处理

日期处理

// 这2个DateFormatter仅仅为了举例子而写的,具体细节根据自己需求而定
private let date1Fmt: DateFormatter = {
let fmt = DateFormatter()
fmt.dateFormat = "yyyy-MM-dd"
return fmt
}() private let date2Fmt: DateFormatter = {
let fmt = DateFormatter()
fmt.dateFormat = "yyyy-MM-dd HH:mm:ss.SSS"
return fmt
}() struct Student: Convertible {
var date1: Date?
var date2: NSDate? // 实现kj_modelValue方法
// 会传入属性`property`以及这个属性对应的JSON值`jsonValue`
// 返回值是你希望最后设置到模型属性上的值
// 如果返回`jsonValue`,代表不做任何事,交给KakaJSON内部处理
// 如果返回`nil`,代表忽略这个属性,KakaJSON不会给这个属性设值(属性会保留它的默认值)
func kj_modelValue(from jsonValue: Any?, _ property: Property) -> Any? {
switch property.name { // 如果jsonValue是字符串,就直接转成Date
case "date1": return (jsonValue as? String).flatMap(date1Fmt.date) // 如果jsonValue是字符串,就直接转成Date
// 由于NSDate与Date之间是可以桥接转换的,所以返回Date给NSDate属性也是没有问题的
case "date2": return (jsonValue as? String).flatMap(date2Fmt.date) default: return jsonValue }
}
} let date1 = "2008-09-09"
let date2 = "2011-11-12 14:20:30.888" let json: [String: Any] = [
"date1": date1,
"date2": date2
] let student = json.kj.model(Student.self)
// 将Date\NSDate转回字符串进行比较
XCTAssert(student.date1.flatMap(date1Fmt.string) == date1)
XCTAssert(student.date2.flatMap(date2Fmt.string) == date2)

不确定类型

// 有时候服务器返回的某个字段的内容类型可能是不确定的
// 客户端可以先标记为Any类型或者AnyObject类型或者协议类型等不确定类型 struct Person: Convertible {
var name: String = ""
var pet: Any? func kj_modelValue(from jsonValue: Any?, _ property: Property) -> Any? {
// 如果不是`pet`属性,就按照默认处理
if property.name != "pet" { return jsonValue }
// 如果是`pet`属性,并且`jsonValue`是个字典,就转换为`Dog`模型实例
// 具体判断逻辑可以根据实际开发需求而定
return (jsonValue as? [String: Any])?.kj.model(Dog.self)
}
} struct Dog: Convertible {
var name: String = ""
var weight: Double = 0.0
} let json: [String: Any] = [
"name": "Jack",
"pet": ["name": "Wang", "weight": 109.5]
] let person = json.kj.model(Person.self)
XCTAssert(person.name == "Jack") let pet = person.pet as? Dog
XCTAssert(pet?.name == "Wang")
XCTAssert(pet?.weight == 109.5) /*---------------------------------------------*/ class Book: Convertible {
var name: String = ""
var price: Double = 0.0
required init() {}
} struct Person: Convertible {
var name: String = ""
// [AnyObject]、[Convertible]、NSArray、NSMutableArray
var books: [Any]? func kj_modelValue(from jsonValue: Any?,
_ property: Property) -> Any? {
if property.name != "books" { return jsonValue }
// if books is `NSMutableArray`, neet convert `Array` to `NSMutableArray`
// because `Array` to `NSMutableArray` is not a bridging conversion
return (jsonValue as? [Any])?.kj.modelArray(Book.self)
}
} let name = "Jack"
let books = [
(name: "Fast C++", price: 666),
(name: "Data Structure And Algorithm", price: 1666)
] let json: [String: Any] = [
"name": name,
"books": [
["name": books[0].name, "price": books[0].price],
["name": books[1].name, "price": books[1].price]
]
] let person = json.kj.model(Person.self)
XCTAssert(person.name == name) XCTAssert(person.books?.count == books.count) let book0 = person.books?[0] as? Book
XCTAssert(book0?.name == books[0].name)
XCTAssert(book0?.price == Double(books[0].price)) let book1 = person.books?[1] as? Book
XCTAssert(book1?.name == books[1].name)
XCTAssert(book1?.price == Double(books[1].price))

其他例子

struct Student: Convertible {
var age: Int = 0
var name: String = "" func kj_modelValue(from jsonValue: Any?, _ property: Property) -> Any? {
switch property.name { // 如果`age`属性的`jsonValue`是整数,就加上5
case "age": return (jsonValue as? Int).flatMap { $0 + 5 } // 如果`name `属性的`jsonValue`是字符串,就在前面加上`kj_`
case "name": return (jsonValue as? String).flatMap { "kj_" + $0 } default: return jsonValue }
}
} let json: [String: Any] = [
"age": 10,
"name": "Jack"
] let student = json.kj.model(Student.self)
XCTAssert(student.age == 15)
XCTAssert(student.name == "kj_Jack")

其他实现思路

// 关于值过滤、自定义值处理的逻辑,也可以在模型转换完毕之后进行

struct Student: Convertible {
var age: Int = 0
var name: String = "" // 实现`kj_didConvertToModel`方法,在这里修改转换之后的属性值
mutating func kj_didConvertToModel(from json: [String: Any]) {
age += 5
name = "kj_" + name
}
} let json: [String: Any] = [
"age": 10,
"name": "Jack"
] let student = json.kj.model(Student.self)
XCTAssert(student.age == 15)
XCTAssert(student.name == "kj_Jack")

最后的提示

  • kj_modelValue也支持ConvertibleConfig配置,用法类似于kj_modelKey,参考第三篇文章

【KakaJSON手册】04_JSON转Model_04_值过滤的更多相关文章

  1. 【KakaJSON手册】01_JSON转Model_01_基本用法

    在iOS开发中,后台返回的数据大多是JSON格式,对应地会被网络框架层解析成Swift中的Dictionary.Array.由于数据类型的复杂.字段的繁多,直接使用Dictionary.Array会比 ...

  2. 【KakaJSON手册】06_Model转JSON

    前面的文章介绍了如何利用KakaJSON进行JSON转Model,从这篇文章开始介绍如何将Model转成JSON 生成JSON和JSONString struct Car: Convertible { ...

  3. 【KakaJSON手册】02_JSON转Model_02_数据类型

    由于JSON格式的能表达的数据类型是比较有限的,所以服务器返回的JSON数据有时无法自动转换成客户端想要的数据类型. 比如服务器返回的时间可能是个毫秒数1565480696,但客户端想要的是Date类 ...

  4. 【KakaJSON手册】03_JSON转Model_03_key处理

    有时候,服务器返回的JSON数据的key跟客户端模型的属性名可能不一致,比如客户端遵守驼峰规范叫做nickName,而服务器端返回的JSON可能叫做nick_name.这时候为了保证数据转换成功,就需 ...

  5. 如何在django-filter中用choice field 的 value 值过滤对象

    如果我们有这样一个model: class IPInfoModel(models.Model): TYPE_INTRANET = 1 TYPE_INTERNET = 2 IP_TYPES = ( (T ...

  6. 【KakaJSON手册】05_JSON转Model_05_动态模型

    在上一篇文章中提到:有时候服务器返回的某个字段的内容类型可能是不确定的 当时给出的解决方案是实现kk_modelValue或者kk_didConvertToModel方法,根据实际需求自定义JSON的 ...

  7. 【KakaJSON手册】08_其他用法

    除了完成JSON和Model的转换之外,KakaJSON内部还有很多实用的功能,有些也开放为public接口了 遍历属性 struct Cat { var age: Int = 0 let name: ...

  8. 【KakaJSON手册】07_Coding_归档_解档

    KakaJSON可以只用一行代码将常用数据进行归档\解档 后面代码中会用到 file 文件路径 // 文件路径(String或者URL都可以) let file = "/Users/mj/D ...

  9. Python数据科学手册-Pandas:数据取值与选择

    Numpy数组取值 切片[:,1:5], 掩码操作arr[arr>0], 花哨的索引 arr[0, [1,5]],Pandas的操作类似 Series数据选择方法 Series对象与一维Nump ...

随机推荐

  1. Bzoj 1040 [ZJOI2008]骑士 题解

    1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 5368  Solved: 2044[Submit][Status ...

  2. MyBatis从入门到精通(2):MyBatis XML方式的基本用法

    本章将通过完成权限管理的常见业务来学习 MyBatis XML方式的基本用法 2.1一个简单的权限控制需求 权限管理的需求: 一个用户拥有若干角色,一个角色拥有若干权限,权限就是对某个模块资源的某种操 ...

  3. vue组件之间的传值——中央事件总线与跨组件之间的通信($attrs、$listeners)

    vue组件之间的通信有很多种方式,最常用到的就是父子组件之间的传值,但是当项目工程比较大的时候,就会出现兄弟组件之间的传值,跨级组件之间的传值.不可否认,这些都可以类似父子组件一级一级的转换传递,但是 ...

  4. Jira 使用手册

    Date Revision version Description author 2018-06-14 V1.0.0 Isaac Zhang 2018-06-22 V1.0.1 1,添加git提交操作 ...

  5. 描述符\get/set/delete,init/new/call,元类

    6.23 自我总结 1.描述符__get__,__set__,__delete__ 描述符是什么:描述符本质就是一个新式类,在这个新式类中,至少实现了____get__(),__set__(),__d ...

  6. [PTA] 数据结构与算法题目集 6-11 先序输出叶结点

    //函数PreorderPrintLeaves应按照先序遍历的顺序输出给定二叉树BT的叶结点,格式为一个空格跟着一个字符. void PreorderPrintLeaves(BinTree BT) { ...

  7. C#编程之IList、List和ArrayList

    额...今天看了半天Ilist<T>和List<T>的区别,然后惊奇的发现使用IList<T>还是List<T>对我的项目来说没有区别...  在C#中 ...

  8. Java EE.Servlet.生成响应

    Servlet的核心职责就是根据客户端的请求生成动态响应. 1.编码类型 2.流操作(下载文件) servlet支持两种格式的输入/输出流.一种是字符输入输出流.另一种是字节输入输出流. 3.重定向

  9. Java中注释的使用

    Java 中注释有三种类型:单行注释.多行注释.文档注释 我们可以通过 javadoc 命令从文档注释(/**aa*/)中提取内容,生成程序的 API 帮助文档. 打开首页,查看下生成的 API 文档 ...

  10. ironic+nova详解

    ironic+nova详解 说明: Openstack 的安装步骤省略,按照社区的文档即可搭建出一套相对稳定的使用环境.本文档基于Newton版本. 假设现在已经有一套可用的Newton环境, 以下的 ...