参考文章:
MVVM in Swift
MVC 与MVVM的区别
在MVC下ViewController中常常会包含以下代码:
- 网络请求
- 状态同步
- 用户交互
- 视图布局
- 图片缓存
- 后台数据处理等
而在MVVM中:
- 模型的作用和MVC相同;
- ViewModels 只与model交互;
- ViewControllers不能直接与model交互,只能通过ViewModel来操作视图;
- 视图(View)只能和ViewController交互,提供交互事件。
MVC和MVVM之间的关键区别在于:
- 有一个新的ViewModel类,但是并不是你的项目里多了个ViewModel你就能说你用的是MVVM了;
- ViewController没有访问Model的权限了。
看图

ViewModel的职责是处理所有的逻辑,ViewController与ViewModel进行“绑定”,当ViewModel拿到最新数据是会直接呈现在ViewController上,“绑定”操作可以通过KVO或FRP(函数式反应编程)来实现。
下面来通过个例子来帮助读者理解
- 用户交互
现在控制器顶端有个Segment,我想通过枚举来实现Index的使cell重新排序,代码长这个样子
enum SwitchValues: Int {
case Grid = 0
case LeastBids
case MostBids
case HighestCurrentBid
case LowestCurrentBid
case Alphabetical
}
那么问题来了,这个枚举应该放到ViewController里呢,还是放到ViewModel中呢?既然要实现排序,貌似是逻辑处理,那就是ViewModel了,可是index的改变似乎只是改变了视图的布局,和逻辑关系也不太大,这样放ViewController里好像也是可以的哦!这下尴尬了!笔者的建议是放到ViewModel中,布局是通过ViewModel来发送信号改变的,ViewController通过信号来改变布局,这样放到ViewModel中貌似更合理一些!(由于原文作者的代码已过时,所以就不贴了,感兴趣的可以在查看文章开头的原文链接)
- ViewModel的结构
对于各位iOS开发者来说,关于MVVM和FRP最关心的一个问题就是如何实现ViewModel和ViewController之间的绑定!ViewController需要及时知道model的变化,这里有两种实现方法:
- 将其定义为ViewModel的属性,通过KVO来监测其值的改变,在OC中的ReactiveObj-C有个RAC的宏,可以很方便的实现这个功能!但是Swift没有宏啊,所以推荐第二种方式。
- 使用信号、队列等,通过framework中的方法来实现同步操作,在RxSwift中有许多绑定的方法。
- 用户交互
在ViewController中是无法直接访问Model的,当用户点击了button以后,怎样将model呈现出来呢,当然用ViewModel的属性是一种方法,还有一种方法需要用到Swift中的闭包,这个东西真是方便啊!
还是看图

具体流程是这样的:
- 用户点击cell
- ViewController实现indexPath选中的回调
- ViewController告知ViewModel选中的indexPath
- ViewModel查找需要的Model
- ViewModel实现展现细节传递给之前初始化的闭包
- 闭包展示Model
饶了这么一大圈就只实现了一个点击功能,貌似有点繁琐!但是当你习惯了MVVM这种开发模式之后,发现其利远大于弊,在下口拙,无法传递这种便利,各位大大自行研究后,欢迎交流!
最后,加个Demo,有问题欢迎交流!
网友评论