谈谈MVC和MVVM

作者: JamesYu | 来源:发表于2015-09-12 17:07 被阅读39025次

工作到现在,对代码稍微有点认识,if else 会写那么点。今天就和大家谈谈所理解的MVC,以及现在比较流行的MVVM。首先我们应该明白,计算机实现一个功能核心代码就那么点。也许我们经常会听到对于同一个问题,菜鸟的实现的真的就是几行,可是大牛却多出了很多文件。这个是为什么?因为大牛的“经验”比较多。。。这里面最主要的目的就是为了维护和可扩展。在设计模式里面,如果你能遵循单一原则,你的代码就已经很好了。

MVC

做iOS开发,一直被教导一定要按MVC模式开发。可是MVC到底是神马?至少在我工作初也不懂是咋回事!如果网上一搜就是千篇一律的M是数据模型,V是视图,C是控制器。然后巴拉巴拉的讲他们之间是怎么通信的。然并无卵用!iOS的MVC展现形式还是有点特殊的,特别是controller和view紧密在一起,controller还必须负责view的展示。在服务器端,view根据model直接生成HTML,然后直接扔给浏览器去渲染和展示,通过Ajax或者js post告诉服务器controller view的响应事件,controller真的做的只是业务数据的处理,出来的结果其实还是数据,根本没有去做UI相关的事情。也可以说服务器根本就没有view。。但它有个浏览器,帮他管理着view和controller之间的交互。而在我们iOS中,controller可不仅仅只是处理数据了,还负责view的管理以及事件的传递。 MVC本质就是将数据展示和数据进行进行隔离,提高代码的复用性和扩展性。好吧,我也说点并无什么卵用的。

看看斯坦福老爷爷的一张图:

这就是我们所认识的MVC。我们可以看到,Controller可以和Model通信,也可以和View进行通信。继续看Controller和Model的关系,绿色的箭头代表Controller可以直接进行对Model进行访问,也就是说Model对于Controller来说就是透明的。但是Model并不知道Controller是谁。如果Model发生了变化,那么就通过Notification和KVO的方式传递给Controller。同样的Controller和View之间也是这种关系,View对Controller来说就是 透明的。Controller可以直接根据Model决定View的展示。View如果接受响应事件则通过delegate,target-action,block等方式告诉Controller的状态变化。Controller进行业务的处理,然后再控制View的展示。

到这里你会发现Model和View并不能直接的进行通信,都必须通过Controller。那这样Model和View就是相互独立的。View只负责页面的展示,Model只是数据的存储,那么也就达到了解耦和重用的目的。

说这么多不如几行代码来的实在。我们以一个简单的例子来看下:

我们假设苹果根据买iPhone的人给予不同的优惠,学生优惠20%,it民工优惠50%,其他不优惠。

//客户类

typedef NS_ENUM(NSInteger, CustomerType) {
    CustomerTypeStudent,
    CustomerTypeiT,
    CustomerTypeOther,
};

@interface Customer : NSObject

@property (nonatomic, assign) CustomerType customerType;

@end

//iPhone类

@interface iPhone : NSObject

@property (nonatomic, strong) NSString *name;
@property (nonatomic, strong) NSString *price;

@end

//VC类

@interface ViewController ()

@property (nonatomic, strong) iPhone *iphone;
@property (nonatomic, strong) Customer *customer;

@property (weak, nonatomic) IBOutlet UILabel *lblName;
@property (weak, nonatomic) IBOutlet UILabel *lblPrice;
@property (weak, nonatomic) IBOutlet UILabel *lblDiscount;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    self.title = @"手机优惠";

    self.lblName.text = self.iphone.name;
    self.lblPrice.text = self.iphone.price;

    if (self.customer.customerType == CustomerTypeStudent) {
        self.lblDiscount.text = @"优惠20%";
    }
    else if (self.customer.customerType == CustomerTypeiT) {
        self.lblDiscount.text = @"优惠50%";
    }
    else {
        self.lblDiscount.text = @"没有优惠";
    }
}

@end

这个就是我们最正常的开发,我们的逻辑都是在vc里面写的。这样写有木有错呢?一点没错,controller本来就是用来处理业务的。由于这个例子比较简单,只是做了个优惠判断,所以我们看不出有啥坏处。有点开发经验的都知道,如果业务复杂起来,再加上其他乱七八糟的验证,controller就会变得很大,越来越难以维护。这个也是MVC比较明显的缺点。

MVVM

既然controller越来越臃肿,越来越难以维护,我们怎么去优化和瘦身呢?回头再仔细看看我们所谓的业务逻辑,是干什么的?无非就是根据几个数据得出一个数据用来控制view的显示。比如展示的是什么文案,按钮能不能响应,页面能不能跳转等等。那MVVM就干了这件事,帮忙分担一下controller里面的部分业务逻辑。MVVM更合理的应该叫做MV-CM。

这个时候,controller将不再直接和真实的model进行绑定了,而通过ViewModel,viewModel进行持有真实的Model。

我们来看看刚刚那例子怎么修改:

//新建一个viewModel
//.h文件

@interface viewModel : NSObject

@property (nonatomic, strong) NSString *name;
@property (nonatomic, strong) NSString *price;
@property (nonatomic, strong) NSString *discount;

- (id)initWithCustomer:(Customer *)customer iphone:(iPhone *)iphone;

@end

//.m文件

@interface viewModel ()

@property (nonatomic, strong) iPhone *iphone;
@property (nonatomic, strong) Customer *customer;

@end

@implementation viewModel

- (id)initWithCustomer:(Customer *)customer iphone:(iPhone *)iphone
{
    if (self = [super init]) {
        _customer = customer;
        _iphone = iphone;
        [self bindData];
    }
    return self;
}

- (void)bindData
{
    self.name = _iphone.name;
    self.price = _iphone.price;

    if (self.customer.customerType == CustomerTypeStudent) {
        self.discount = @"优惠20%";
    }
    else if (self.customer.customerType == CustomerTypeiT) {
        self.discount = @"优惠50%";
    }
    else {
        self.discount = @"没有优惠";
    }
}

@end

//VC

@interface ViewController ()

@property (nonatomic, strong) viewModel *viewModel;

@property (weak, nonatomic) IBOutlet UILabel *lblName;
@property (weak, nonatomic) IBOutlet UILabel *lblPrice;
@property (weak, nonatomic) IBOutlet UILabel *lblDiscount;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    self.title = @"手机优惠";

    self.lblName.text = self.viewModel.name;
    self.lblPrice.text = self.viewModel.price;
    self.lblDiscount.text = self.viewModel.discount;

}

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

看到修改完的代码,你会发现VC里面已经省去了不少的代码。一切都和viewModel进行交流。这里我只是展示一个最简单的数据展示,如果有其他响应事件,是需要viewModel开放方法来进行处理的,并要通知VC处理结果的。

关于MVVM的优点:

  • 方便测试

    在MVC下,Controller基本是无法测试的,里面混杂了个各种逻辑,而且分散在不同的地方。有了MVVM我们就可以测试里面的viewModel,来验证我们的处理结果对不对(Xcode7的测试已经越来越完善了)。

  • 便于代码的移植

    比如iOS里面有iPhone版本和iPad版本,除了交互展示不一样外,业务逻辑的model是一致的。这样,我们就可以以很小的代价去开发另一个app。(以前做公司iPad的时候就深深感觉到,全部在VC里面是多么的痛苦和重新开发一个没有啥区别)。

  • 兼容MVC

    MVVM是MVC的一个升级版,目前的MVC也可以很快的转换到MVVM这个模式。VC可以省去一大部分展示逻辑。

缺点:

  • 类会增多

    每个VC都附带一个viewModel,类的数量*2

  • viewModel会越来越庞大

    我们把逻辑给了viewModel,那势必Model也会变得很复杂,里面的属性和方法越来越多。可能重写的方法比较多,因为涉及到一些数据的转换以及和controller之间的通信。

  • 调用复杂度增加

    由于数据都是从viewModel来,想想突然来了一个新人,一看代码,不知道真实的模型是谁。比如常用tableview的数据源,一般都是一个数组,如果不断的通过viewModel去取,沟通上没有那么直接。况且每封一层,意味着要写很多代码去融合他们的转换。

最后说下ReactiveCocoa这个框架,这个虽然和MVVM经常一起出现,这个框架主要是帮我们实现model和view的绑定机制。后面会有文章来介绍它。

相关文章

  • MVC - MVVM 是什么

    MVC - MVVM 是什么 谈谈MVC模式 - 阮一峰 MVC,MVP 和 MVVM 的图示 - 阮一峰 MVC...

  • 谈谈MVC和MVVM

    工作到现在,对代码稍微有点认识,if else 会写那么点。今天就和大家谈谈所理解的MVC,以及现在比较流行的MV...

  • 重学巩固你的Vuejs知识(上)

    前沿 谈谈你对MVC、MVP和MVVM的理解? https://github.com/webVueBlog/int...

  • 6、设计模式

    Q:谈谈MVC、MVP和MVVM,好在哪里,不好在哪里? https://www.jianshu.com/writ...

  • MVC、MVVM

    MVC和MVVM都是用来分离model和view的MVC:model、view、controller MVVM:m...

  • Android 综合技术面试题

    1.1 请谈谈你对 MVC、MVP、MVVM、MVI的理解? MVC Model:主要用于网络请求、数据库、业务逻...

  • MVC、MVP、MVVM模式浅谈

    根据阮一峰老师的谈谈MVC模式 及MVC,MVP 和 MVVM 的图示 总结的思维导图(新建标签页才能看清楚图片)...

  • 3.mvc和mvvm的区别和使用

    1.mvc和mvvm的由来 2.mvvm的概念 3.mvc和mvvm的区别 4.MVVM的实践 1.model层的...

  • 架构模式

    MVC和MVVM详解

  • vue的mvvm原理解析及手写一个

    # 手写vue的mvvm实现原理 ## 1:mvc和mvvm的区别? MVC:modal-view-control...

网友评论

  • selice:看了这个文章,我餐发现刚刚接手这个项目原来是mvvm模式写的
  • f48ab9e63152:ReactiveCocoa 推荐
  • 会跳舞的狮子:你说后面会有文章介绍RAC 现在已经过去 2 年了
  • 故事姥爷:第一句话太谦虚了。
  • c8deab353dd1:前端的表示看不懂ios代码
  • b4d73bf0ea05:简洁明了一下子就懂了,哈哈还是几行代码来的实在,谢谢大佬!
  • 铮铮钱钱:这个不算是mvvm吧
  • 山林间迷雾能不能当障眼法的内容:快出RAC 想问问你们大厂都用这个么
  • UncleFool:请教一下楼主:如果一个Controller包含多个Model,应该针对每个Model创建不同ViewModel,还是针对Controller创建一个ViewModel来集中处理Model?
    UncleFool:@JamesYu 之前对ViewModel的理解是对Model进行处理,做些数据校验、转换、过滤等工作。如文中代码,Controller中包含了两个Model。ViewController是服务于Controller,分担Controller业务逻辑,所以他们是一对一关系。才想明白,谢谢楼主。
    JamesYu:@UncleFool 一个controller对应一个viewModel
  • cdaa77b2f11e:请教一下,从“数据驱动”和“事件驱动”的角度分析,mvvm程序会不会比mvc程序 跑起来更耗资源啊?
    老王技术栈:不管是不是数据驱动还是事件驱动。都会更耗资源吧、毕竟需要更多的类、�
  • __________mo:代码里缺了viewmodel的初始化?initWithCustomer?
    JamesYu:@__________mo 多点思考。。东搬西搬的原因是啥?有没有想过
    __________mo:@JamesYu 感觉这些模式就是东搬西搬:sweat:
    JamesYu:@__________mo 外部传进来的
  • aa6b20dfde1a:结合代码【应用】,实在,好!
  • 一个脱离了高级趣味的人e:看了一堆文章,楼主写的最明白!
    talk is cheap, show me the code
    配合着代码展示清晰明了,谢谢楼主分享!!
    3557efe1a951:看了这么多回复,就你回复的最新
  • biyuhuaping:看了一堆文章,我觉得你讲的MVVM最容易理解。
  • Captain_XR:感谢分享,棒棒哒
  • CodingWord:很不错
  • cc1d10fb31fd:如果vc的view model有不只一个,以功能区分就好了,不过文件就更多
  • 16f10d6e5e11:看不懂
  • 郑伟的菜园子:看了一堆文章,我觉得你讲的MVVM理解是对的。
  • 嗨_我是大鑫啊:感觉比MVC更难维护了呢????
    iOS_小胜:复杂逻辑的业务模块可以用MVVM,一般的MVC就够用了,类多意味着功能分工明确,只要你明白这些类职责,不会更难维护
    JamesYu:@BigXin 比较累赘,如果项目严格按这个来。类就要多一倍。。很多业务也不需要都这样。。
  • 哲人王:如果我要写个按钮事件,如点击提交【购买】按钮,然后提交到服务器,那么这个事件和调用方法在MVVM该如何做代码组织呢?即是否是那业务逻辑写在viewModel里,能给个SAMPLE CODE么?谢谢
  • antz:⛽️

本文标题:谈谈MVC和MVVM

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