美文网首页人猿星球IOS开发
面向对象设计之iOS网络库

面向对象设计之iOS网络库

作者: Delpan | 来源:发表于2020-05-13 15:28 被阅读0次

    序言

    本文不是描述一个完整网络库的设计,也不会涉及多少实现原理上的讲解,主视角是放在面向对象的分析与设计上。

    本文的前半部份看起来可能会比较烦琐,因为前半部份的内容是抛开过于抽象的理论概念来讲述的,抽象的理论概念可以指引我们怎么高效率处理相似的问题,但抽象的理论概念比较晦涩,所以我们会站在更基础的视角来看问题,讲述会偏基础。

    下面会先给出一个精简后的真实需求,然后我们通过一步步的分析与设计解决需求问题。市面上已经有很多很成熟的第三方库,相同的需求我们用第三方库是怎么处理的。任何的工作都要考虑效率问题,分析与设计也一样,不同的分析与设计过程效率也会不一样。

    废话不多说,全世界埋位,Action!!Demo

    1. 需求

    客户端与服务端基于HTTP协议进行交互,每个请求报文都会有公共的参数和报文头字段,请求报文Body的序列化方式可能是JSON或者URLEncode,响应报文Body的反序列化方式也可能是JSON或者URLEncode,反序列化后得到的数据会存在多种结构和字段名。

    2. 分析与设计

    确定需求后,我们需要对需求进行分析与设计,这个过程是包括,抽象关键对象,抽象对象之间的共性,设计对象的关系与交互,确定实现方式以及边界问题。

    2.1 提取关键抽象

    首先从需求中我们得知客户端与服务端是基于HTTP协议进行交互,而HTTP的基本交互就是在一个HTTP会话中进行一问一答,客户端向服务端发送请求,服务端对请求进行响应。从这个交互过程中我们可以抽象出的对象是HTTP会话HTTPSession,请求Request与响应Response

    每个请求报文都会有公共的参数和报文头字段,而这些公共信息可以是编译期决定的,也可以是运行时动态添加或修改,所以我们需要一个对象来管理这些公共信息RequestGeneralInfo

    请求报文Body的序列化方式可能是JSONURLEncode,既然是存在多种Body的序列化方式,所以这里可以直接抽象出两个的对象RequestBodyJSONSerializerRequestBodyURLEncodeSerializer

    响应报文Body的反序列化方式也可能是JSONURLEncode,在这里我们可以抽象出两个的对象ResponseBodyJSONSerializerResponseBodyURLEncodeSerializer

    2.2 对象抽象视图

    我们从需求中提取出关键对象后,接下来要定义这些对象的抽象视图。这些关键对象都是基于某种动机抽象出来的,我们可以根据抽象动机来定义这些对象的初始行为(也可以叫接口或操作)和属性。首先来看HTTPSession这个对象,它是用来抽象HTTP会话,所以它的行为和属性可以根据HTTP会话来定义。

    HTTPSession2.png

    HTTP会话的交互大概就像上图那样,发送一个Request,接收一个Response并回反馈,所以HTTPSession对象具有发送Request的行为,以及接收并回调Response的行为,虽然HTTP会话是一问一答的方式进行交互,但是HTTP会话可以提前缓冲一组Request,只是这些Request是按顺序来发送并接收响应的,所以有当前Request集合的属性,接收到Response后我们需要把信息回调给相应的客户对象,所以有客户对象集合的属性,根据这些基础的信息我们可以得到HTTPSession对象的初始行为和属性以及基础抽象视图。

    HTTPSessionView.png

    Request对象和Response对象分别抽象了HTTP请求报文和响应报文,所以可以由此根据来定义它们的初始行为和属性。

    RequestAndResponse2.png

    RequestGeneralInfo对象只是用来存储公共参数和报文头字段,所以行为和属性的定义比较简单。

    RequestGeneralInfo3.png

    虽然请求报文和响应报文Body的序列化对象有多个,但由于动机明确,职责单一,都只是用来做参数的序列化工作,所以行为和属性比较简单。

    RequestBodySerializer3.png
    ResponseBodySerializer2.png

    2.3 抽象交互过程

    我们定义了关键对象的初始行为和属性,以及这些对象的基础抽象视图。这些对象是否已经可以满足我们的需求?我们尝试使用这些对象来解决需求问题。

    这里解释一下什么是客户对象,什么是服务对象,对象A向对象B发送消息,对象B响应消息并提供相应服务,在这个场景下,对象A就是客户对象,对象B就是服务对象。举个例子,ViewController对象向Model对象发送获取数据的消息,Model对象接收消息并返回数据给ViewController对象,那ViewController对象就是客户对象,而Model对象就是服务对象。

    Scene1_1.png

    从上图我们可以看到,客户对象跟这些基础对象一顿交互后,确实已经解决了基本的需求问题。但同时我们也发现,对于所有需要进行HTTP交互的客户对象而言,这一烦琐的交互过程都是必不可少的,一百个客户对象就会有一百份相同的交互代码。有一百份相同的代码看上去好像也没什么问题,顶多不就是冗余代码多一点,以后优化一下交互过程,瞬间代码量减少了一个量级,今年安装包瘦身的KPI又完成了,从这个角度来看,CV大法是真的香!

    我们现在来考虑后期维护的问题,假设参与这个交互过程的其中一个服务对象的行为发生了变化,怎么办?还能怎么办,加班改啊。对象行为的改变是后期维护比较常见的情况,通过修改一百份相同的代码来进行维护,这种方式本身就非常低效。

    现在所遇到的问题是存在重复的交互过程,我们可以尝试把这个交互过程抽象成一个操作对象HTTPJSONOperation来复用这个交互过程。我们再进行上面的交互时,只需要去创建一个HTTPJSONOperation对象,让HTTPJSONOperation对象代替客户对象去完成烦琐的交互,最后把结果返回给相应的客户对象即可。

    HTTPJSONOperation2.png

    从上图可以看到,我们即减少了代码的冗余量,同时当某些服务对象的行为发生变化时,我们只需要改变HTTPJSONOperation对象的实现即可,这对客户对象并没有产生什么影响,所以客户对象的稳定性就提高了。

    虽然抽象出HTTPJSONOperation对象后,整个交互过程被简化了,稳定性也挺高了,但我们回头看一下需求“请求报文Body的序列化方式可能是JSON或者URLEncode,响应报文Body的序列化方式也可能是JSON或者URLEncode”时就会发现另一个问题,HTTPJSONOperation对象只能处理JSON类型的交互,而从需求当中我们得知,我们即有JSON类型的交互,也有URLEncode类型的交互,所以我们现在要处理URLEncode类型的交互。

    HTTPURLEncodeOperation2.png

    我们可以用相同的方式对URLEncode类型的交互进行抽象,抽象出HTTPURLEncodeOperation对象后,我们再进行URLEncode类型的交互时只需要创建一个HTTPURLEncodeOperation对象即可。

    Scene4_1.png
    Scene5_1.png

    从上面的两个交互过程我们发现,无论是JSON操作对象还是URLEncode操作对象都是从RequestGeneralInfo对象中获取公共信息,然后再设置Request对象,所以这个交互过程是重复的,我们可以用相同的方式来处理,抽象出一个RequestSetOperation操作对象来复用这个交互过程,但这里再插入一个RequestSetOperation操作对象,那参与整个交互过程的对象又增加了,交互过程也比现在要复杂一些。除了操作对象这种处理方式,我们是否还有别的方式可以解决重复交互过程的问题?

    我们观察一下这个重复的交互过程和参与的对象就会发现,RequestGeneralInfo对象的职责以及存储的信息都是跟Request对象相关,所以我们可以为RequestGeneralInfo对象新添加一个行为,把设置Request对象的工作交给RequestGeneralInfo对象完成。

    RequestGeneralInfo4.png
    Scene6_1.png

    2.4 共性抽象

    在分析与设计过程中,我们常常会抽象出一些职责非常相似的对象,这些对象会存在一定的共性,我们可以尝试抽象这些共性来获得更好的动态性。

    通过观察HTTPJSONOperation对象和HTTPURLEncodeOperation对象后就会发现,这两个对象除了Body的序列化/反序列化方式不同以外,无论是行为,属性还是交互方式都是一样的。也就是说,这两个对象的实现大部份都是重复的,我们是否有办法合并这两个对象来进一步减少代码的冗余量呢?

    合并两个相似的对象主要要考虑的问题是,怎么兼容不同的部份。一种方式是把所有不同的内容都包含进来,通过枚举类型区分并选择相应的实现方式。

    HTTPOperation6.png

    这种方式看上去好像也没什么问题,但当你尝试去扩展一种新的交互类型时,这种方式的缺点就很明显了。首先添加新的枚举值XML,然后添加XML的序列化/反序列化对象为属性,接下来就是在需要序列化/反序列化时,通过if else的方式来选择相应的序列化/反序列化对象。首先这是一种硬编码的扩展方式,其次用这种方式扩展需要打开原始文件以及修改原来已经稳定的实现代码。这些为什么是缺点,这些缺点会带来什么问题,这部份会在后面的内容讲解,这里先假设这种方式是坑爹的。

    2.4.1 抽象协议

    除了上面这种方式,还有没有别的方式可以用来兼容不同的部份呢?我们可以先回过头来观察一下HTTPOperation对象与RequestBodyJSONSerializer对象以及RequestBodyURLEncodeSerializer对象的交互情况。

    HTTPOperationAndRequestBodySerializer.png

    从上图的交互流程可以发现,HTTPOperation对象无论是跟RequestBodyJSONSerializer对象交互,还是跟RequestBodyURLEncodeSerializer对象交互,都是向序列化对象发送同一消息,而序列化对象也是以相同的方式返回结果给HTTPOperation对象,也就是说,这两个序列化对象有相同的行为,而这个相同的行为被HTTPOperation对象所依赖,所以我们可以尝试把这个相同行为做高一层级的抽象(或者叫向上抽象),抽象成更一般性的协议(或者叫接口)RequestBodySerializerRequestBodyJSONSerializer类与RequestBodyURLEncodeSerializer类继承(或者叫引入)并实现协议。

    RequestBodySerializer4.png

    HTTPOperation对象依赖协议提供的服务。

    HTTPOperationNew2.png

    从上图可以看到,HTTPOperation对象只知道RequestBodySerializer协议提供的服务,但HTTPOperation对象并不知道具体是哪个对象提供的,当我们想进行某类型的交互时,只需要给HTTPOperation对象传相应类型的RequestBodySerializer对象即可。扩展新的交互方式时,我们只需要把新抽象的对象实现RequestBodySerializer协议,我们就可以用新的方式进行交互,而这种扩展并不需要打开和修改HTTPOperation对象的实现代码,就可以复用HTTPOperation对象的整套交互流程,这种动态扩展后面会详细讲解。

    ResponseBodyJSONSerializerResponseBodyURLEncodeSerializer对象可以用相同的方式处理,抽象出ResponseBodySerializer协议,这里就不再描述推导过程。

    2.4.2 进一步抽象交互过程

    Scene2_2.png

    通过抽象协议,我们合并了操作对象,且操作对象的复用性更强了,但整个交互过程看上去还是比较烦琐,我们是否可以简化这个流程,或者减少一些交互的对象呢?

    我们回过头来观察一下整个交互过程可以发现,RequestGeneralInfo对象和RequestBodySerializer对象具有相似的职责,都是跟Request对象的序列化相关,只是各自负责的部份不同,我们是否可以利用这一点来做些什么呢?

    我们可以尝试合并这两个对象的职责,现在把RequestGeneralInfo对象的职责添加到RequestBodySerializer对象上。

    RequestBodySerializerAndInfo2.png

    我们得到了一个新的对象RequestSerializer,这个对象负责管理公共信息以及将相关信息序列化,现在用RequestSerializer对象来进行交互。

    HTTPSessionAndRequestSerializer.png

    从上图可以看到,使用RequestSerializer对象后,整个交互过程确实是简化了,RequestSerializer对象完成了设置以及序列化Request对象。但我们回头看需求时就会发现这种方式又来带了另外一些问题,“请求报文Body的序列化方式可能是JSON或者URLEncode”,为了可以进行JSON类型和URLEncode类型的交互,所以我们需要在程序运行时同时持有RequestBodyJSONSerializer对象和RequestBodyURLEncodeSerializer对象,这两个对象都保存了公共信息,也就是在程序运行时相同的数据有两份,当公共信息需要更新时,需要同时更新这两个对象。合并这两个对象的职责似乎并不是一个好的选择,所以我们需要再回过头来观察交互过程。

    Scene2_2.png

    从上图可以看到,RequestGeneralInfo对象的工作和RequestBodySerializer对象的工作存在一定关联,HTTPOperation对象先跟RequestGeneralInfo对象交互,设置Request对象,然后RequestBodySerializer对象再去序列化Request对象,也就是说,RequestBodySerializer对象的工作是建立在RequestGeneralInfo对象产物的基础上进行的,所以我们可以尝试把RequestGeneralInfo对象作为RequestBodySerializer对象的一部份进行对象组合。

    RequestBodySerializerAndGeneralInfo.png

    HTTPOperation对象需要对Request对象进行序列化时,发送序列化消息给RequestBodySerializer对象,RequestBodySerializer对象再发送设置Request对象的消息给RequestGeneralInfo对象,当RequestGeneralInfo对象完成设置Request对象时,RequestBodySerializer对象再序列化Request对象,并把结果返回给HTTPOperation对象。

    Scene7_1.png

    从上图可以看到,站在HTTPOperation对象的视角来看,整个交互过程是简化了,因为HTTPOperation对象需要交互的对象减少了,同时在程序运行时也只需存在一个RequestGeneralInfo对象,即公共信息在程序运行时只有一份。

    观察RequestGeneralInfo对象和RequestBodySerializer对象可以发现它们存在相同的行为,而它们之间的关系又是紧密的,所以我们可以抽象它们的共性。我们在前面也发现了,RequestBodySerializer对象的工作是建立在RequestGeneralInfo对象的产物的基础上进行的,我们可以抽象这一概念,即RequestBodySerializer对象作为一个序列化对象,它的工作是建立在另一个序列化对象的产物的基础上进行的。基于这些概念我们可以抽象出一个新的类层次结构。

    RequestBodySerializerNew2.png

    我们现在来看看这些对象运行时的关系。

    HTTPOperationRelation1_1.png
    Scene3_2.png

    从上图可以看到,我们进一步简化了HTTPOperation对象的交互过程,同时也把这个交互过程的抽象级别又提高了一层。对于HTTPOperation对象而言,它只知道RequestSerializer对象是用来序列化Request对象,至于RequestSerializer对象是一个组合对象还是单一对象,这对于HTTPOperation对象是透明的。

    RequestSerializerLevel.png

    我们可以基于前面总结出来的抽象概念以及RequestSerializer协议做更多更复杂的扩展,可以设计复杂的序列化过程。在任一分支上的所有RequestSerializer对象都是通过在前一个RequestSerializer对象产物的基础上做进一步的序列化工作,在这里我们就得到了一种可以动态扩展的模式。

    到此为止,我们基本上已经处理完请求报文的问题,接下来处理响应报文的问题。

    2.4.3 个体与整体的抽象

    ResponseBodySerializerNew2.png

    上图是我们在前面得到的ResponseBodySerializer协议,响应报文的处理跟请求报文的处理有些不同,我们可以从响应报文头字段知道Body是用什么方式序列化,我们用相同的方式进行反序列化即可得到数据,所以我们不必将响应报文的反序列化方式跟HTTP交互一一捆绑起来。正常情况下,服务端也可以根据请求报文头字段的信息对Body进行反序列化,如果服务端愿意配合客户端的话。

    HTTPOperationAndResponseBodySerializer.png

    我们需要做的是统一处理响应报文Body的反序列化,所以我们可以基于ResponseBodySerializer协议抽象出一个新的具体对象。

    ResponseBodyComposeSerializer.png

    这个对象是一个聚合体,我们可以动态向ResponseBodyComposeSerializer对象添加所需要的ResponseBodySerializer对象。当HTTPOperation对象向ResponseBodyComposeSerializer对象发送反序列化消息时,ResponseBodyComposeSerializer对象把消息转发给当前保存的ResponseBodySerializer对象处理即可,这样我们就保证了ResponseBodyComposeSerializer对象可动态扩展性和稳定性,我们又得到了一种可以动态扩展的模式。

    2.5 操作对象管理

    通过进一步的分析与设计,我们不仅解决了重复交互过程的问题,还得到了一个可以动态扩展,复用性强的HTTPOperation对象。我们把交互过程抽象以后,这个交互过程在未来进行调整也不会影响客户对象,程序的稳定性也得到了提升。

    ClientAndHTTPOperation2.png

    通过观察客户对象与HTTPOperation对象的交互,我们又有新的发现,由于HTTPOperation对象是一个抽象化的交互过程,所以客户对象需要给HTTPOperation对象传递相应的实体对象,让这些实体对象参与这个交互过程,而这个传递过程对于每一个客户对象来说都是重复的,而且从前面的分析与设计的经验我们知道,一个对象对另一个对象行为的依赖越强,交互越紧密,它们之间的耦合度就越高,耦合度就越高后期扩展和维护就越难,很容易会出现改一个对象把整个系统都改崩了,俗称“牵一发动全身”。

    我们发现客户对象不仅负责传递实体对象,还负责创建并管理这些实体对象,并且还要管理HTTPOperation对象,但我们也发现这几个行为是相关的,所以我们可以尝试抽象一个对象来完成这些操作,对象抽象的动机也很明确,就是负责创建并管理相关的实体对象,负责设置并管理HTTPOperation对象。

    HTTPOperationManager3.png

    我们可以在程序运行时创建一个生命周期与程序生命相同的HTTPOperationManager对象,这个对象用来存储最常用到的相关实体对象,这样我们就可以用这个对象来创建并管理最常用的HTTPOperation对象,我们也可以在获取到HTTPOperation对象后进行特殊的处理。我们不仅减少了客户对象的依赖,同时要解决了重复交互过程以及管理的问题。

    2.6 实现方式与边界问题

    HTTPNetwork3.png

    经过一轮又一轮的分析与设计,我们得到了一个基础框架HTTPNetwork,以及基于这个基础框架所做的扩展,但我们在前面做的工作只是把需求变现,设计了对象之间的关系,接下来我们考虑对象的实现以及对象之间的边界问题。正常情况下,这个过程可能会对现有的框架和架构做一些微调整,可能会适当调整对象的抽象视图,也可能会抽象出一些新的对象。

    我们先来看HTTPOperation对象,HTTPOperation对象是抽象了HTTP的交互过程,这里需要考虑的是,这个HTTP交互过程是同步还是异步,是单线程环境还是多线程环境,运行环境由哪个对象提供,开始交互后是否还能改变参与交互过程的对象,交互完成后回调方式以及回调环境,存在多种回调方式还是单一回调方式,交互是否可以取消。

    HTTPOperationManager对象是用来管理HTTPOperation对象和参与HTTP交互的相关对象,初始化HTTPOperationManager对象后还能不能改变参与HTTP交互的相关对象,如果能改变,当前已运行或待运行的HTTPOperation对象是否更新相应的对象,HTTPOperationManager对象用方式来存储HTTPOperation对象,HTTPOperationManager对象是否为HTTPOperation对象提供运行环境,是否需要控制HTTPOperation对象运行的最大并发数,等等

    上面提出了很多问题,这里就不给出什么答案了,如果把所有的问题和解决方式的优缺点都列清清楚楚楚,那真是写到明年都写不完,用一话广东话来形容,写到蚊都训啦。

    实现方式的选择可能产生的影响,举个例子,HTTPOperationManager对象需要控制HTTPOperation对象运行的最大并发数,而最大并发数是可设置的,所以我们需要为HTTPOperationManager对象添加新的行为,这样就改变了对象的抽象视图。为了方便存储和管理HTTPOperation对象,我们可以抽象一个新的数据结构对象来完成这一工作。

    当我们做完分析与设计,考虑清楚实现方式和边界问题后,我们就可以开始动手写代码,不过这个时候的写代码基本上就是打字了。

    3. 设计模式

    平时跟朋友,同事或者网友沟通交流过程中发现一个现象,有很多开发者对设计模式和设计原则这些抽象理论概念很熟,但面对真实需求的时候就是不知道怎么动手做设计,这好像是一个比较普遍的现象。

    0A43E36146EA609FC11528E29B292249.gif

    鲁迅曾经说过“世界上本没有设计模式,相似的对象交互及关系用多了,便成了设计模式”。设计模式是前人在摸索对象之间的关系及交互,通过一个又一个的实例,不断实践,把一个个相似的场景总结出来的高级抽象概念,所以你纯粹去背设计模式和设计原则,做不出设计不挺正常的。我没有经济学基础,就去听了一次巴菲特的讲座,我就能成为一个经济学家,那真是有鬼了。

    在前面的分析与设计过程中,我并没有提及设计模式,设计原则或者一些过于抽象的理论概念,甚至刻意去回避这些东西,纯粹站在面向对象的角度进行分析与设计,最终我们也可以设计出一个基础框架,也可以得到一个解决方案。我们在摸索对象之间的关系及交互时所总结出来的模式,也可能就是现在的设计模式。

    RequestBodySerializerNew2.png
    HTTPOperationRelation1_1.png

    我们回头看“2.4.2-进一步抽象交互过程”。RequestGeneralInfo对象和RequestBodySerializer对象具有相同的行为,RequestBodySerializer对象的工作是基于RequestGeneralInfo对象的产物进行的,也就是说RequestBodySerializer对象动态扩展了RequestGeneralInfo对象,最后我们总结出来的模式就是现有的装饰模式。我们在“2.4.3-个体与整体的抽象”总结出来的就是现有的组合模式。

    很多时候在你考虑对象之间,类之间的关系及交互时,你不一定可以找到相应的设计模式用,这个时候就要考验你面向对象的基础了。抽象理论概念很熟但不知道怎么动手做设计,这个坑我以前也踩过,我分享一下我自己的学习经历(嗯,接下来开始编故事了),希望对你有帮助。

    我最初想学怎么做面向对象设计的时候,也是直接看的设计模式和设计原则,在学习的头一年我基本上都能把整本《GoF23》和相关的设计原则背出来,如果单纯讨论设计模式和设计原则这些抽象理论概念,我能吹得头头是道,很像是那么一回事,当我去维护一些现有的库,现有的框架的时候,我也能在原有的基础上套用一些设计模式去做点事情,也因此当时就产生了幻觉,“哦,原来设计模式和设计原则就是这么一回事”。

    但当我面对一份陌生的需求,要根据实际情况从0到1做设计的时候就懵逼了,完全都不知道怎么动手。这就好比你去打Dota2,可以自由选英雄,选装备,选等级和技能,然后就打一场后期的团战,然后你发现其中一个队友无论是切入时机,走位,技能释放,装备使用都很像那么一回事,光看这场团战的操作和意识,你都以为他是个大屌,当你跟他组队从开局开始打,你发现他就是一个沙雕,选个Lina上来一级升被动,然后对线期对面6级他才2级,全场被追着提款。

    后来我从一个国外的架构师所分享的学习线图中发现,他推荐的设计模式相关的书是《Head First》和《GoF23》,但无论《Head First》还是《GoF23》,都是从另一本书指向过来的,也就是说那本书是基础,那本书是《面向对象分析与设计》,看到这本书的时候我比较疑惑,面向对象技术无论是在学校学到的,还是工作后了解到的,无非不就是封装,继承和多态这三个概念,还有别的什么东西吗?当我花了几个月认认真真把这本书啃下来才发现,还是too young too simple!封装,继承和多态那都只是面向对象技术的冰山一角。

    接下来经历了一个比较痛苦的时期,因为要转变思维方式,告别单纯的封装,继承和多态,从头再来。这就像你吹了十年的Saxophone,然后突然发现你的气息和嘴型都是错的,我TMD(挺猛的)......然后通过不断实践,慢慢把思维方式转变过来,相对稳定之后再回头看设计模式和设计原则就清晰多了,而且理解也不一样了,虽然我已经背不出来。

    EE50041A4F96A496B6AC69E598BD6337.jpg

    4. 第三方库的处理方式

    AFNetworking应该算是iOS最常见的网络库了,我们现在尝试用AFNetworking来解决我们的需求。HTTP交互由AFNetworking中的AFHTTPSessionManager完成。

    响应反序列化AFHTTPResponseSerializer跟我们上面分析设计的一样,也是通过组合模式实现的,AFCompoundResponseSerializer作为反序列化对象的聚合体,保存各种具体的AFHTTPResponseSerializer对象。虽然AFNetworking没有URLEncode类型的反序列化对象,但我们可以继承AFHTTPResponseSerializer实现一个AFURLEncodeResponseSerializer即可。

    AFHTTPResponseSerializer.png

    跟我们上面分析与设计不同的是,请求序列化AFHTTPRequestSerializer是用策略模式实现的,而我们的RequestSerializer是用装饰模式实现的,设计模式不同就意味着扩展方式不同,实现方式以及对象的责职和边界都不一样。

    AFHTTPRequestSerializer.png

    我们是把HTTP交互抽象成一个HTTPOperation对象,这样我们就可以根据需求来决定参与HTTP交互的对象。在这一点上,AFHTTPSessionManagerAFHTTPRequestSerializer是做不到的,虽然我们可以动态设置AFHTTPSessionManager相应的AFHTTPRequestSerializer对象,但我们不能根据需求对每一次的HTTP交互进行设置,AFHTTPSessionManager并没有相应的行为可以解决这一问题。

    一种解决方式是,我们可以创建两个AFHTTPSessionManager对象,一个持有AFHTTPRequestSerializer对象,一个持有AFJSONRequestSerializer对象,但这种方式我们在上面也推导过,当需要更新请求报文头字段时,必须同时更新AFHTTPRequestSerializer对象和AFJSONRequestSerializer对象。

    另一种解决方式是,我们可以创建一个AFHTTPSessionManager分类,在分类添加可以根据需求对每一次的HTTP交互进行设置的行为,但这种方式只能用AFURLSessionManager对象的行为来实现,而且AFHTTPSessionManager整个类基本上是多余了。

    AFNetworking在这种场景下所表现出来的灵活性是有一点欠缺的,当然这里并不是说AFNetworking不好,只是场景不同,AFNetworking预设的场景是服务端可以根据请求报文的信息进行反序列化工作的,这种预设的场景更常规,所以AFNetworking的请求序列化用策略模式很合理,只是我们实际遇到了奇葩的需求。

    5. 扩展

    到此为止,我们还有一个需求没有解决,“响应报文Body的反序列化方式也可能是JSON或者URLEncode,反序列化后得到的数据会存在多种结构和字段名”,多种数据结构和字段名所产生的影响是显而易见的。

    Scene8_2.png

    从上图可以看到,当ViewController对象接收到响应数据后,ViewController对象需要对数据结构进行识别,还要识别不同的字段名,这些识别工作和处理对于每个ViewController对象都是一样的,最简单的解决方式就是CV大法,每个ViewController都复制一份处理,CV大法的一个问题是会产生大量冗余代码,CV大法最大的问题还不是这个,会出现多种数据结构和字段名这也侧面反映出服务端是比较随便的,后期再次更改数据结构和字段名是大概率发生的事情,所以当数据结构和字段名再次更改时,对ViewController的影响就很严重了,这有多恶心就不用多说了。

    一种解决方式是,抽象出一种固定的数据结构,HTTPOperation对象返回这种固定的数据结构,这就要求HTTPOperation对象在接收到响应报文后对数据进行识别和处理,多种数据结构和字段名,所以我们需要抽象多个处理对象。

    ResponseDataStructSerializer2.png

    从上图可以看到,我们抽象了这些对象的共性,和响应反序列化一样,我们可以统一处理这些不同的数据结构和字段名,所以这里也可以应用组合模式。

    从职责上来看,ResponseDataStructSerializer也属于响应反序列化的一部份,而ResponseDataStructSerializer对象的工作是在ResponseBodySerializer对象的产物的基础上进行的,所以这里可以应用装饰模式对ResponseDataStructSerializer对象和ResponseBodySerializer对象进行组合。

    ResponseSerializer2.png

    这里所做的扩展对原有的HTTPNetwork基础框架不会产生影响。这样,HTTPOperation对象就可以返回固定的数据结构给ViewController对象。

    Scene9_1.png

    后期再更改数据结构和字段名也不会对ViewController对象产生影响,因为ViewController对象每次拿到的都是统一标准的数据结构。

    6. 面向对象分析与设计的过程

    在前面的分析与设计过程中,我是分了几个阶段来进行,各个阶段关注不同的事情,每个阶段都在上一个阶段的产物的基础上进行。我早期分析与设计的过程不是这样的,我早期的方式是,先总结需求,然后根据需求抽取关键抽象,接下来对象之间的关系和实现方式我会一并考虑,最早的方式就更乱了,直接上来就是想实现问题。那为什么现在会把分析与设计的过程拆得更细,分更多的阶段进行?主要是因为效率和稳定性。

    6.1 复杂性与稳定性

    我们现在假设需求相对复杂一些,能抽象出三四十个关键对象,面对这么多对象,光去考虑这些对象的关系,这已经是一件复杂的事情,还要去考虑这些对象的交互,考虑交互的时候还会不停地修改对象的抽象视图,调整对象的关系,这又是一件复杂的事情,同时还要考虑这些对象的实现,考虑对象的实现可能又会修改对象的抽象视图,可能会添加一些新的对象。

    在这个过程中大概率会出现的情况是,你可能对所有的东西都只是大概是这样子,都是一些模凌两可的想法,或者你大概想好了一些对象的实现方式,但通过进一步的考虑,你可能会合并一些对象,或者在交互过程中发现要拆一些对象,那你刚刚想好的实现基本上就是废了。在你考虑实现的同时去考虑对象之间的交互,很容易就会把两个原本应该独立的对象,实现上做了依赖,当最后你想去改一下某一个对象的实现,可能整份设计都崩了,或者硬挤出来的设计在真实场景中应用时,可能会发现到处都不合适,最后只能推倒重来。这样的稳定性就会很差。

    6.2 效率

    如果目标不明确,准确率低,稳定性差,你就需要不断推倒重来,那你要做出一份合理的设计所花费的时间就会很多,而且你面对一个混乱的局面,你可能连动手的方向都没有。

    孔子曾经说过:“人的大脑在同一时间内能关注和处理的事情是有限的”。常规解决复杂问题的方式就是,把大问题拆成一个个可以处理的小问题,再把一个个小问题解决,这样大问题也就解决了。当你把大的过程,拆成多个阶段,每个阶段只专注做一件事件,目标很明确,所以你得到的产物就很稳定,推倒重来的可能性也大大降低了,而且你最后得到的设计的准确率也会很高。

    7. 最后

    通过一轮的努力,我们得到了一个简单的网络库,虽然跟完整的网络库还有很长的距离,但我们已经迈出了第一步。我面对的真实需求要比这个复杂得多,情况也很特殊。

    正常情况下,项目当中的不同模块应该是完全独立的,域名也是独立的,而我们项目当中所有的模块都是用同一个域名,所以我们会很依赖长连接。复用长连接能在一定程序上提升网络性能,不过我们的服务端狗就狗在,就连文件下载都是在同一个域名。我最早是用NSURLSession来交互的,NSURLSession虽然有配置对象,配置对象也有很多属性,但这些属性基本上都是参考值,不是关键值,所以NSURLSession在同一时间同一域名开多少条TCP连接完全看“心情”来的,经常出现的情况是,虽然有多个任务,但NSURLSession只开一条TCP连接交互,这就导致同一时间内,如果有大文件占用了这条TCP连接,后面的数据交互基本就废了,无奈之下最后我是用GCDSocket做的交互,自己写模型管理的TCP连接。

    92BC81975DB06545B0A23325A03C1E90.png

    情况特殊也没什么好说的,都是在填服务端的坑,一般情况下能用NSURLSession就用NSURLSession,毕竟效率和稳定性摆在那里,这种东西也没什么好炫技的,又不是用的API越基础越好,我看着我自己写的处理模型真的像陀屎一样。

    想在短短的一篇技术文中全面讲述面向对象分析与设计也是不现实的事,最后我推荐一些看过且觉得好的书,《面向对象分析与设计》,《面向对象的思考过程》,《Head First设计模式》,《GoF23》,《设计之美》,《恰如其分的软件架构》。

    相关文章

      网友评论

        本文标题:面向对象设计之iOS网络库

        本文链接:https://www.haomeiwen.com/subject/iavzghtx.html