上一个项目使用到了ReactiveCocoa+MVVM+AFNetworking+FMDB框架设计,从最初的尝试,到后来不断思考和学习,现在对这样一个整体设计还是有了一定了理解与心得。在此与大家分享下。

本文将不再过多的描述ReactiveCocoaMVVMFMDB的使用细节。关于ReactiveCocoa,我有一篇实用案例的博客:

http://www.brighttj.com/ios/ios-reactivecocoa-utility-demo.html

文章介绍的更多的是我对这个框架设计的理解,而不是具体代码逻辑的讲解。关于代码逻辑,我会在Demo中给出详细的注释,本文Demo下载地址:

https://github.com/saitjr/ReactiveCocoa-MVVM-AFNetworking-FMDB.git

环境信息:

Mac OS X 10.11

Xcode 7.0.1

iOS 9.0.1

ReactiveCocoa 2.4.7

AFNetworking 2.6.1

FMDB 2.5

MJExtension 2.5.14


正文

工程目录

先来谈谈工程目录吧,如图:

工程目录

1. 【Module】+【Model】

在这个目录中,比较核心的是【Module】与【Model】,他们组成了整个MVVM框架。

【Module】与【Model】均包含【Base】,其中有BaseModelBaseViewModelBaseViewController。在开发中,我还是习惯无论是否需要基类,都去写一个。难免开发之初就考虑到,也难免之后需求会变更。

2. 【Interface】接口

这是借鉴了Java中的接口思想,目的是为了统一方法名。例如里面的SQLInterface.h文件,就是一个对数据进行CRUD操作的protocol,并且可以规定里面的方法是否必须实现。

3. 【Configuration】配置

对项目的一些基本配置,如基本宏定义、常量、通知名,亦或是Cell的identifier。宏定义中一般包含项目基本属性,如:主色调、常用方法等。

在提供的Demo中,我将SQL语句放在了SQL.h中,是因为SQL只有一个文件在引用,其中的定义方式是:

static NSString * const selectArticleSQL = @"SELECT * FROM article";

NotificationNames.h会在大部分文件中用到,所以使用UIKIT_EXTERN定义为了全局变量:

-----.h
UIKIT_EXTERN NSString * const LoadAllNotification;
-----.m
NSString * const LoadAllNotification = @"LoadAllNotification";

5. 【Category】类目

项目中没有打包的类目,例如给三方或系统类新增的一些方法。

RAC+MVVM

RAC+MVVM在【Module】和【Model】这两个目录中进行实现。在这之中,MVVM是框架思想,RAC只是辅助而已。

一、MVVM

之所以采用MVVM,而不是MVC,也得益于MVVM的一大特点,就是减轻C层的负担,毕竟以前的C层完全就是百宝箱,什么样的代码都写在里面。

对于MVVM我的理解和拆分是这样的:

1. Model与View

这一层和MVC中的Model、View含义相同。

2. ViewModel

这一层主要作用是将以前写在ViewController中的数据处理放在ViewModel中,如:网络请求、数据缓存、无法直接展示的数据处理(如NSNumber这类的,就在VM中处理成NSString,然后V层直接用,而不是在V层中处理)。

二、Demo中VC的设计

图形结合源码应该能看个大概。

VC设计

三、自定义cell的设计(可延伸至自定义View的设计)

因为介绍的是VC,所以这里再单独说一下HomePageCell这个自定义cell的设计。

因为想到实际项目中,可能会有比较复杂的cell,所以Demo中写的是一个比较完整的设计方式(如果单单看这个Demo的话,这个自定义cell太简单,没必要有一个单独的VM,有点过度设计)。

cell中的构思是,cell有一个CellVM来管理cell中要显示的数据,CellVM来自于VC中,dataSource数组。处理方式具体是:在网络请求完成以后,将字典->模型,然后通过模型,初始化CellVM,然后将CellVM放入dataSource数组。

Cell实现部分代码如下:

@implementation HomePageCell

- (void)awakeFromNib {
[super awakeFromNib]; [self setupSignal];
} - (void)setupSignal { @weakify(self);
[RACObserve(self, viewModel) subscribeNext:^(HomePageCellViewModel *viewModel) { @strongify(self);
self.textLabel.text = viewModel.titleText;
self.detailTextLabel.text = viewModel.authorText;
}];
} @end

AFNetworking

前几天,AFNetworking升级到了3.0。将以前基于NSURLConnection的API,全都改成了NSURLSession,关于更新的详情,可以看AFNetworking 3.0迁移指南这篇文章。

在页面销毁、重新请求等情况下,需要将还在队列中的请求取消,以免占用资源。

考虑到这一点,结合我的网络请求在VM发送,权衡再三,在BaseViewModel的基础上,再进行了一次封装——RequestViewModel。这个VM有AFHTTPSessionManager类的属性,一个该属性的懒加载和一个在dealloc中取消请求方法。

在以前使用MVC的时候,我会对AFNetworking进行再次的封装,这样更像是一个MVCS的设计,目的是防止VC过重,现在把这部分代码扔在了VM中,看起来还好,所以并没有对AFNetworking再次封装。关于以前的设计方式,可以看这篇文章:

网络请求框架封装

FMDB

FMDB提供了一种线程安全的模式,在这之中维护这一个串行队列。

1. 初始化

初始化方式参考了开源项目MVVMReactiveCocoa,作者采用了类目的形式给了一个单例:

@implementation FMDatabaseQueue (Extension)

+ (instancetype)shareInstense {

    static FMDatabaseQueue *queue = nil;

    static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
queue = [FMDatabaseQueue databaseQueueWithPath:DB_PATH];
}); return queue;
} @end

2. 统一接口

关于CRUD方法的定义,借鉴了Java Interface的设计:

3. SQL语句

根据实际情况,SQL语句参数可以采用?的形式,也可以采用:keyword的形式。

INSERT INTO myTable VALUES (?, ?, ?)
INSERT INTO myTable VALUES (:id, :name, :value)

关于数据库的创建与更新,我并不建议在Bundle中包含xxx.sql这样的文件,因为他们会在编译后一起打包,用户下载解压就能看到,并不怎么安全。目前我是直接在程序中写的SQL,不知道大家有没有更好的方式。

4. CRUD

在网络请求成功的时候,存入数据。存储是一个批量的操作,建议采用事务inTransaction实现。简单的操作采用inDatabase即可。

FMDatabaseQueue是一个串行队列,并且inTransactioninDatabase都是同步线程,需要注意的是不要在block中执行另一个数据库访问操作,防止线程死锁

最后

每个项目完后,都会有很多收获,有很多东西需要整理总结。写这篇博客的原因有两个:

原因之一:因为我在开发过程中踩了不少坑,可能开发到中途,发现框架设计不好。框架如何设计,并没有一个标准答案,而且设计思想,还需要在不断实践中得出,所以每次总结,是为了给自己看,也是为了帮到其他有同样困扰的朋友。

原因之二:也正是因为我不知道框架到底怎么样,所以写出来,让大家看看,都多多提出建议。谢谢。

参考

1. 雷纯峰的开源项目MVVMReactiveCocoa,这个项目给我的启发很大,在此谢谢作者,也为开源点赞

2. iOS大型项目开发漫谈

3. iOS应用架构谈 网络设计方面

【iOS】小项目框架设计(ReactiveCocoa+MVVM+AFNetworking+FMDB)的更多相关文章

  1. iOS开发项目之MVC与MVVM

    MVC MVC,Model-View-Controller,我们从这个古老而经典的设计模式入手.采用 MVC 这个架构的最大的优点在于其概念简单,易于理解,几乎任何一个程序员都会有所了解,几乎每一所计 ...

  2. 我的2016年终总结(PF项目框架设计心得分享 2.0rc)

    在无数的日夜里,熬出了多少的黑眼圈,致勤勤恳恳工作的各位朋友与自己.每到了年末的时候总想写的什么,主要是为了回顾以往一年里到底做了什么,这便是年终总结的主要意义.在此我将要总结的是和我在技术层面上成长 ...

  3. 简单搭建iOS开发项目框架

    今天我们来谈谈如何搭建框架,框架需要做一些什么. 第一步:找到我们的目标我们的目标是让其他开发人员拿到手后即可写页面,不再需要考虑其他的问题. 第二步:我们需要做哪些东西各位跟着我一步一步来进行. 假 ...

  4. 01 Taro_Mall 开源多端小程序框架设计

    项目介绍 Taro_Mall是一款多端开源在线商城应用程序,后台是基于litemall基础上进行开发,前端采用Taro框架编写,现已全部完成小程序和h5移动端,后续会对APP,淘宝,头条,百度小程序进 ...

  5. 我的2017年终总结(PF项目框架设计心得分享 1.0rc new)

    一晃眼又过去了一年,在这一年里尽管有许多不如意的事,却阻挡不了我前进的脚步.先用一句话来总结去年一年的状态,那就是“无休无止的忙碌”.而这样的忙碌状态对我来说是不可取的,因为匮乏的忙碌只能让头脑处于一 ...

  6. ios学习笔记图片+图片解释(c语言 oc语言 ios控件 ios小项目 ios小功能 swift都有而且笔记完整喔)

    下面是目录其中ios文件夹包括了大部分ios控件的介绍和演示,swift的时完整版,可以学习完swift(这个看的是swift刚出来一周的视频截图,可能有点赶,但是完整),c语言和oc语言的也可以完整 ...

  7. ReactiveCocoa框架下的MVVM模式解读

    记录一些MVVM文章中关于ReactiveCocoa的代码: 实例一:带有分页的文章列表,根据文章类别过滤出文章的列表,可以进入文章详细页面 1:YFBlogListViewModel 首先了解关于列 ...

  8. iOS 从零到一搭建组件化项目框架

    随着公司业务需求的不断迭代发展,工程的代码量和业务逻辑也越来越多,原始的开发模式和架构已经无法满足我们的业务发展速度了,这时我们就需要将原始项目进行一次重构大手术了.这时我们应该很清晰这次手术的动刀口 ...

  9. iOS 模仿一个小项目,总结一下里边的模块

      ManoBoo:  参考链接:http://www.jianshu.com/p/fd4c46c31508  这个小的项目是参考ManoBoo的简书的,链接在上方,自己在仿做的过程中,也离不开Man ...

随机推荐

  1. Vim魔法堂:认识快捷键绑定

    Brief 习惯在VS上按<F5>来编译运行程序,刚用上VIM上就觉得无比的麻烦,而随着对VIM的学习我们分阶段的简化这一操作 1. 退出VIM,在shell下编译&&运行 ...

  2. HtmlAgilityPack 处理通配的contains

    //选择不包含class属性的节点 var result = node.SelectNodes(".//span[not(@class)]"); //选择不包含class和id属性 ...

  3. 从C#到Objective-C,循序渐进学习苹果开发(4)--代码块(block)和错误异常处理的理解

    本随笔系列主要介绍从一个Windows平台从事C#开发到Mac平台苹果开发的一系列感想和体验历程,本系列文章是在起步阶段逐步积累的,希望带给大家更好,更真实的转换历程体验.本文继续上一篇随笔<从 ...

  4. 走进异步世界-犯傻也值得分享:ConfigureAwait(false)使用经验分享

    在上周解决“博客程序异步化改造之后遭遇的性能问题”的过程中,我们干了一件自以为很有成就感的事——在表现层(MVC与WebForms)将所有使用await的地方都加上了ConfigureAwait(fa ...

  5. MVC-自定义HttpModule处理

    HttpModule是向实现类提供模块初始化和处置事件. 当一个HTTP请求到达HttpModule时,整个ASP.NET Framework系统还并没有对这个HTTP请求做任何处理,也就是说此时对于 ...

  6. mssql游标实现递归查询

    表值函数 存储过程

  7. Java --HashMap源码解析

    兴趣所致研究一下HashMap的源码,写下自己的理解,基于JDK1.8. 本文大概分析HashMap的put(),get(),resize()三个方法. 首先让我们来看看put()方法. public ...

  8. 关于线上的bug什么时候修复的思考

    这里系统专门指的是那种用户量大的系统,比如有几百万或者上千万的注册会员.因为小系统因为用户量少,不存在这种思考,考虑有时候是多余的.另外还有内部系统,给自己公司内部人员使用的,即便是出现了问题,也不会 ...

  9. 文件并发(日志处理)--队列--Redis+Log4Net

    多线程操作同一个文件时会出现并发问题.解决的一个办法就是给文件加锁(lock),但是这样的话,一个线程操作文件时,其它的都得等待,这样的话性能非常差.另外一个解决方案,就是先将数据放在队列中,然后开启 ...

  10. HTML学习总结

    首先,我们要问,什么是html?官方的解释是:超文本标记语言.什么意思呢?简单的来说,就是一种用来制作网页的特殊语言.那么,什么是网页呢?我们说,网页是一个在浏览器窗口下显示的页面,实质上是一个文档. ...