SwiftyJSON源代码学习(三)

概述

上文已经分析了SwiftyJSON的数据处理流程,此时已经可以通过一个Any类型的参数构造一个解包后拥有类型信息和相关rawValue的JSON结构体,接下来要做的就是设计一个统一的、便捷的接口供用户获取结构体内部数据。
在此之前我们先来更清晰的梳理一下,假如输入为二进制流数据data,其内容为:
{ "imageName": "Lina", "urls": ["www.anExample.com", "www.anotherExample.com"] }
经过JSON结构体的构造方法let json = JSON(data)后,生成的json实例的type.dictionary,解包后的数据实际储存在rawDictionary属性中。要获取其中的具体数据,以下标的形式json["imageName"]无疑是最方便的,而且与原生的风格一致。

面向协议编程

在Swift中有一个名为subscript特性,用于提供下标形式的访问方式。因为对于JSON来说,最外层要么是Dictionary,要么是Array,也就是说是必定是集合类型,所以我们先把JSON结构体扩展为Swift的集合类型,即实现Swift.Collection协议。因为JSON结构体的内部数据可能为多种类型,要扩展为集合,就要先定义一种统一的索引方式。而作为索引,其本身必须是可比较的,也就是必须实现Comparable协议。

关于索引IndexComparable协议

对于JSON结构体来说,我们定义了7种类型,但是作为索引,只需要针对.array.dictionary.null,其他类型本身就是一个实体,没有索引的概念(.string本身是有索引的,不过我们不需要把字符串给拆成字符)。所以声明一个枚举,把上述三种情况统一为一个Index,并实现Comparable协议。

Comparable协议实际就是重载比较运算符,如==<等,让两个同类型的实例可以进行比较,协议的概念和实现的逻辑都比较简单:

关于Collection协议

Collection协议可以为实现该协议的类型提供几乎全部集合常用的特性,如通过下标获取集合元素,for...in遍历集合元素,count isEmpty indices等常用属性,元素操作、距离、切片、迭代器等常用方法。

遵守Collection协议至少必须满足以下三点:

  • startIndexendIndex属性用来定义元素的起始
  • subscript特性用来通过下标获取集合内部元素
  • index(after:)方法用来确定元素的排列顺序
    理解起来也很清晰明了,一个集合通过subscript特性把索引和内部的元素绑定,并且确定了起始位置的索引和其他索引的后继,那么这个集合内部的元素就已经全部确定且可知的了。

而由于ArrayDictionary都已经是集合类型了,我们定义的Index只是通过枚举对几种情况进行了统一,那么实现协议的时候也只需要对Index类型进行判断,然后调用ArrayDictionary本身的相关实现即可:

关于下标的更优设计

此时我们就可以通过下标来访问JSON结构体了,还以文章开头的json为例,要获取imageName可以这样json[.dictionary("imageName")]。这可能和想像中的不太一样,下标还必须是枚举且明确其类型,用起来是相当的不方便,所以我们需要设计一种更方便的方法来确定下标。

经过观察可以发现,Array的下标是Int类型,而Dictionary的下标是String类型,那么我们可以实现新的subscript,通过参数的类型来调用相对应的实现。

那么首先分别实现IntString类型参数的subscript,逻辑比较简单,判断类型后,如果索引是合法的则返回索引相对应元素的JSON结构体实例,否则生成相应的错误并赋值到空JSON结构然后返回。

困难在于如何把两者整合起来,SwiftyJSON的解决方案是,定义一个协议JSONSubscriptType,让IntString都去实现该协议,这样参数的类型就统一了。然后再进行判断、调用相应的subscript实现

这样我们终于可以开心的以json["imageName"]这么愉快的方式来获取JSON内部数据了,但是如果我们来取一下第一个url,问题又来了,这种复杂结构的数据又应该怎么获取呢?

SwiftyJSON提供了两种方案,一种是以数组参数的形式,一种是不定参数列表的形式,两种方式除了参数表现形式不同,其实现逻辑是相同的。在get逻辑中,通过依次调用上面的subscript来获取最终的结果,在set逻辑中,如果下标个数是0直接返回,是1个通过上面的subscript去设置,如果多于1个,以递归的方式,逐层获取内部数据,直到剩下最后一层下标,赋值,然后返回。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容