UI:一个IOS工程的标准框架
来自cocachina 的写法 参考
声明:本文来自互联网,非本人原创,仅供参考学习使用。
我的iOS工程结构
接下来,我就简单介绍下我做iOS项目时使用的工程结构。首先要说的是,这只是我的工程结构,并不是规范,或许它存在很多问题和不规范的地方,我只是把它分享出来,给大家提供一个参考,也希望收到大家的一些反馈来帮助我改进!
项目结构
下图是我做iOS项目的一个常用工程结构,整体模式还是按照MVC的结构,只是在每一层做了一些细分处理,下面就简单介绍下。
iOS工程中没有像Java那样非常严格的分包机制,不过在iOS工程中我们也可以通过Group的方式在工程中实现逻辑分包,这样更有利于我们组织和管理代码,使工程结构更清晰和易于理解。在我的工程结构中,主要有如下group:
Application:这个group中放的是AppDelegate和一些系统常量及系统配置文件;
Base:一些基本父类,包括父ViewController和一些公用顶层自定义父类,其他模块的类一般都继承自这里的一些类;
Controller:系统控制层,放置ViewController,均继承于Group Base中的BaseViewController或BaseTableViewController;
View:系统中视图层,由于我比较喜欢通过代码实现界面,所以这里放的都是继承于UIView的视图,我将视图从ViewController中分离出来全部放在这里,这样能保持ViewController的精简;
Model:系统中的实体,通过类来描述系统中的一些角色和业务,同时包含对应这些角色和业务的处理逻辑;
Handler:系统业务逻辑层,负责处理系统复杂业务逻辑,上层调用者是ViewController;
Storage:简单数据存储,主要是一些键值对存储及系统外部文件的存取,包括对NSUserDefault和plist存取的封装;
Network:网络处理层(RTHttpClient),封装了基于AFNetworking的网络处理层,通过block实现处理结果的回调,上层调用者是Handler层;
Database:数据层,封装基于FMDB的sqlite数据库存取和管理(RTDatabaseHelper),对外提供基于Model层对象的调用接口,封装对数据的存储过程。
Utils:系统工具类(AppUtils),主要放置一些系统常用工具类;
Categories:类别,对现有系统类和自定义类的扩展;
Resource:资源库,包括图片,plist文件等;
以上是对我的工程结构中各个group的介绍,通过以下登录模块的系统类图,可以比较直观的看到这种工程结构的全貌。
整体来看分为三大块,黄色区域的模型和业务逻辑层(M),蓝色区域的视图层(V),红色区域的视图控制器层(C),其中,黄色区域实现了对业务逻辑和数据处理的封装,对应他们的上层ViewController,可以实现非常简单的接口调用,将业务复杂性从ViewController中抽离出来,通过模块化的方式,保证ViewController的可读性和可维护性。
保持ViewController简单
往往大家都会抱怨iOS中ViewController写着写着就会越来越臃肿,那时因为随着业务的复杂,功能的增多,所有的逻辑都包含在ViewController中,还包括一些诸如UITableViewDatasource的代理方法,使得ViewController臃肿不堪,可维护性极低,耦合性也很高,为了使ViewController能更简单,便于维护和后续的开发,给ViewController瘦身就显得尤为必要,我的做法主要有三个方面。
1、View视图与ViewController分离
如果你用Storyboard或者xib这是当然的,我比较喜欢手写代码,所以不在ViewController里面嵌入过多的View层代码是保证ViewController简单的方法之一,那么,可以将View部分的代码单独封装到一个继承自UIView的子类当中,然后通过自定义Delegate实现View与ViewController的通信。
2、业务逻辑与ViewController分离
将网络请求处理和复杂的业务逻辑以及数据的存取工作单独放到Handler层,对ViewController只暴露简单的调用接口和通过block或delegate实现的回调,这样不仅能使我们的工程模块化,也能大大降低ViewController的复杂性,就不会出现既包括网络处理又包括数据处理的冗长的ViewController代码了。Handler通过block或delegate将处理完的结果回调给ViewController,ViewController再将结果与View视图层相关联处理,这样就真正起到了MVC的作用,整体原则就是,让ViewController只关系和负责处理与它相关的事。
在BaseHandler.h中可以定义一些简单的业务处理规则:
- #import <Foundation/Foundation.h>
- /**
- * Handler处理完成后调用的Block
- */
- typedef void (^CompleteBlock)();
- /**
- * Handler处理成功时调用的Block
- */
- typedef void (^SuccessBlock)(id obj);
- /**
- * Handler处理失败时调用的Block
- */
- typedef void (^FailedBlock)(id obj);
- @interface BaseHandler : NSObject
- /**
- * 获取请求URL
- *
- * @param path
- * @return 拼装好的URL
- */
- + (NSString *)requestUrlWithPath:(NSString *)path;
- @end
在LoginHandler中就可以定义对LoginViewController暴露的调用接口,在LoginHandler中封装负责的网络处理和业务处理逻辑,对LoginViewcontroller来说,只需要调用这个方法并传入对应的UserEntity实体对象和处理成功和失败状态下的回调block就可以了。
- #import "BaseHandler.h"
- #import "UserEntity.h"
- @interface LoginHandler : BaseHandler
- /**
- * 用户登录业务逻辑处理
- *
- * @param user
- * @param success
- * @param failed
- */
- - (void)executeLoginTaskWithUser:(UserEntity *)user success:(SuccessBlock)success failed:(FailedBlock)failed;
- @end
3、Datasource或Delegate与ViewController分离
在iOS开发中经常用到的UITableView包含了一系列的代理方法,这些方法往往也是使得ViewController变长变复杂的元凶之一,那么,将这些Datasource或Delegate分离出来也是行之有效的方法之一,例如,通过自定义Datasource类(实现UITableViewDatasource协议)来将跟UITableView相关的数据源处理代理方法都集中到一个特定的类当中,ViewController只需要设置这个自定义数据源类给UITableView,然后其他的就都可以交给自定义数据源类去处理了。
我参考了Lighter View Controllers上的介绍改进了一个BaseTableViewProtocol,基本上常用的一些场景是可以使用的,不过这个还得不断优化以适应更多的场景,具体的代码我放在Github上了,感兴趣的同学可以去看看,使用方法可以参考上面链接中的介绍,基本类似,我的改进主要是支持对多section的适用。
写在最后
以上是我在开发iOS项目中的一些总结和工程实践,其中肯定还是存在很多问题的,我也在不断寻求改进的方法,也欢迎各路高手给我提出意见和建议。关于这个工程结构的一个简单事例我放在我的Github上了,感兴趣的同学可以去看看RTLibrary-ios。
UI:一个IOS工程的标准框架的更多相关文章
- 【转】iOS超全开源框架、项目和学习资料汇总
iOS超全开源框架.项目和学习资料汇总(1)UI篇iOS超全开源框架.项目和学习资料汇总(2)动画篇iOS超全开源框架.项目和学习资料汇总(3)网络和Model篇iOS超全开源框架.项目和学习资料汇总 ...
- iOS超全开源框架、项目和学习资料汇总--数据库、缓存处理、图像浏览、摄像照相视频音频篇
iOS超全开源框架.项目和学习资料汇总--数据库.缓存处理.图像浏览.摄像照相视频音频篇 感谢:Ming_en_long 的分享 大神超赞的集合,http://www.jianshu.com/p/f3 ...
- iOS超全开源框架、项目和学习资料汇总:UI篇
上下拉刷新控件 1. MJRefresh --仅需一行代码就可以为UITableView或者CollectionView加上下拉刷新或者上拉刷新功能.可以自定义上下拉刷新的文字说明.(推荐) 2. S ...
- 一个iOS 框架介绍:MKNetworkKit
http://blog.csdn.net/kmyhy/article/details/12276287 http://blog.csdn.net/mobailwang/article/details/ ...
- iOS回顾笔记(05) -- 手把手教你封装一个广告轮播图框架
html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...
- 分享一个UI与业务逻辑分层的框架(二)
序言 第一篇讲解了UI与业务逻辑分层的框架(UIMediator)的使用.本篇将说明该框架的原理及代码实现. 整体结构 UI与后台类绑定主要分为UI输入->后台属性,后台属性-UI更新两部分,为 ...
- [HMLY]7.iOS MVVM+RAC 从框架到实战
1.MVVM浅析 MVC是构建iOS App的标准模式,是苹果推荐的一个用来组织代码的权威范式,市面上大部分App都是这样构建的,具体组织模式不细说,iOS入门者都比较了解(虽然不一定能完全去遵守), ...
- 如何用 React Native 创建一个iOS APP?
诚然,React Native 结合了 Web 应用和 Native 应用的优势,可以使用 JavaScript 来开发 iOS 和 Android 原生应用.在 JavaScript 中用 Reac ...
- ReactiveCocoa - iOS开发的新框架
本文转载至 http://www.infoq.com/cn/articles/reactivecocoa-ios-new-develop-framework ReactiveCocoa(其简称为RAC ...
随机推荐
- transition、animation在macbook air上图片动画边缘抖动2
示例: BUG描述: 最近同事一项目中,产品提出在macbook air上,列表图片放大效果边缘出现抖动现象.在retina屏上没有此问题. 调试过程: 1.单独把结构分离.确定是否由其他元素引起. ...
- linux 文件属性、权限、所有人、所属组
Linux命令行模式下,文件还是需要通过ls -l来查看 可以通过ll查看长文件,会有如下类型显示drwxr-xr-x 2 root root 4096 Nov 10 2010 conf 总共有7 ...
- IO模型:同步、异步、阻塞、非阻塞
前言: 在Linux的网络编程中,同步IO(synchronous IO).异步IO(asynchronous IO).阻塞IO(blocking IO).非阻塞IO(non-blocking IO) ...
- 【转载】TCP粘包问题分析和解决(全)
TCP通信粘包问题分析和解决(全) 在socket网络程序中,TCP和UDP分别是面向连接和非面向连接的.因此TCP的socket编程,收发两端(客户端和服务器端)都要有成对的socket,因此,发送 ...
- c#中的多态 c#中的委托
C#中的多态性 相信大家都对面向对象的三个特征封装.继承.多态很熟悉,每个人都能说上一两句,但是大多数都仅仅是知道这些是什么,不知道CLR内部是如何实现的,所以本篇文章主要说说多态性 ...
- 使用 Docker 在 Linux 上托管 ASP.NET Core 应用程序
说在前面 在阅读本文之前,您必须对 Docker 的中涉及的基本概念以及常见命令有一定了解,本文侧重实战,不会对相关概念详述. 同时请确保您本地开发机器已完成如下安装: Docker 18.06 或更 ...
- bzoj 1030: [JSOI2007]文本生成器 (ac自己主动机上的dp)
1030: [JSOI2007]文本生成器 Time Limit: 1 Sec Memory Limit: 162 MB Submit: 2635 Solved: 1090 [id=1030&qu ...
- 如何在外部采用AES-128对HLS的TS流进行加密
http://www.wowza.cn/hls/howToSecureHlsByAes128External
- DCOS之Mesos-DNS介绍
DCOS的Mesos-DNS它主要提供域名解析服务,Mesos-DNS 在DCOS框架中支持服务发现,同意应用程序和服务通过域名系统(DNS)来相互定位.DCOS中的 Mesos-DNS充当的角色和在 ...
- java 提高效率的做法
可供程序利用的资源(内存.CPU时间.网络带宽等)是有限的,优化的目的就是让程序用尽可能少的资源完成预定的任务.优化通常包含两方面的内容:减小代码的体积,提高代码的运行效率.本文讨论的主要是如何提高代 ...