初探MVVM

从接触iOS开发,最开始学的就是MVC框架,无论是优秀成熟的软件,还是苹果的API都是基于MVC模式进行设计开发的。但由于Controller承载了大量定制性的代码,导致很多项目中的Controller庞大臃肿,甚至有人称MVC为Massive View Controller(重量级视图控制器),难以debug。恰巧要进行项目重构,初次尝试使用MVVM的。

我们先回看一下iOS中MVC

MVC.jpg

上图是非常经典的MVC描述的图片,截取自斯坦福白胡子老师的课件中。这张图片很清晰地描述出了在iOS中MVC三者的作用和关系。

Model:模型层,分离出来的数据层,不可主动与View进行通讯,可通过KVO来被Controller监测变化。

View:视图层,展示视图和接受用户活动,可通过Delegate等方法回调用户行为给Controller。

Controller:控制层,逻辑处理层,拥有Model和View,通过KVO和Delegate等等方式来监听Model和View,当某一方有变化的同时做出相应的逻辑处理。

iOS中的MVC.png

什么是MVVM

因为在项目中ViewController根据业务需求代码量越来越大,那么我们想把一些代码抽离出来,放入到一个单独的类中进行,只留下一些关键性的代码在ViewController中,这样可以大大地提高代码可读性。抽离出来的代码所存放的类就是所谓的ViewModel。Controller监视ViewModel的数据变化,并让ViewModel执行一些如(网络请求,数据处理,数据持久化等操作),在数据变化后执行部分代码,达到ViewModel双向绑定的效果。


MVVM的通讯.png

这样看来,其实MVVM和MVC的模式十分相近,只是将C中的多种逻辑拆离出来,达到了代码可读性更高的效果。

实战代码

FHYBaseViewModel.h

//
//  FHYBaseViewModel.h
//  Created by iOS_Fuhanyu on 16/5/17.
//  Copyright © 2016年 iOS_Fuhanyu. All rights reserved.
//

#import <Foundation/Foundation.h>
/**
 *  viewModel请求成功或失败的回调
 *
 *  @param message 服务器返回的消息
 *  @param isError 是否错误
 *  @param task 请求的任务
 *  @param error 错误信息
 */

typedef void(^requestSuccessBlock)(NSString *message, BOOL isError);
typedef void(^requestFailureBlock)(NSURLSessionDataTask *task, NSError       *error);
typedef void(^verifySuccess)(BOOL isSuccess);
typedef void(^messages)(NSString *message, BOOL isError);
/*
 * 所有列表数据都继承此类 子类需重写init(初始化) 方法 和 parseData(解析) 方法
 */
@interface FHYBaseViewModel : NSObject
/**
 *  数据总条数
 */
@property (nonatomic, strong) NSNumber *total;
/**
 *  是否是分页请求
 */
@property (nonatomic, assign) BOOL isPaging;
/**
 *  请求任务
 */
@property (nonatomic, strong) NSURLSessionDataTask *task;

/**
 *  数据源
 */
@property (nonatomic, strong) NSMutableArray *dataList;
/**
 *  列表数据条数
 */
@property (nonatomic, assign, readonly) NSInteger count;

/**
 *  当前页
 */
@property (nonatomic, assign) NSInteger currentPage;
/**
 *  参数
*/
@property (nonatomic, strong) NSMutableDictionary *parameters;
/**
 *  判断是否有更多数据
 */
- (BOOL)isThereMoreData;

/**
 *  当请求当前页失败或者没有数据的时候 当前页要减一
 */
- (void)resetCurrentPage;

- (void)deleteObjectAtIndex:(NSInteger)index;

/**
 *  索引所对应的元素
 */
- (id)objectAtIndex:(NSInteger)index;

/**
 *  元素对应的索引
 */
- (NSInteger)indexOfObject:(id)object;

/**
 *  处理网络请求的返回数据是否正确
 */
- (void)handleResponseObject:(id)responseObject verifySuccess:(verifySuccess)verifySucdess andMessages:(messages)messages;

/**
 *  解析
 */
- (void)parse:(NSDictionary *)dictionary;

/**
 *  取消请求
 */
- (void)cancel;
@end

FHYBaseViewModel.m

    //  FHYBaseViewModel.m
    //  Created by iOS_Fuhanyu on 16/5/17.
    //  Copyright © 2016年 iOS_Fuhanyu. All rights reserved.
    #import "FHYBaseViewModel.h"

    @implementation FHYBaseViewModel
    - (instancetype)init{
        if (self = [super init]) {
            self.isPaging = YES;//yes代表当前借口有分页
            self.currentPage = 1; //当前页
        }
        return self;
    }

    - (NSMutableArray *)dataList{
        if (_dataList == nil) {
           _dataList = [NSMutableArray array];
        }
        return _dataList;
    }

    - (NSInteger)count{
        return _dataList.count;
    }

    - (void)resetCurrentPage{
        if (self.currentPage > 1 && _isPaging) {
            self.currentPage--;
        }
    }

    - (void)deleteObjectAtIndex:(NSInteger)index{
        [_dataList removeObjectAtIndex:index];
    }

    - (NSInteger)indexOfObject:(id)object{
        return [_dataList indexOfObject:object];
    }

    - (id)objectAtIndex:(NSInteger)index{
        return [_dataList objectAtIndex:index];
    }

    - (BOOL)isThereMoreData{
        return _dataList.count < _total.integerValue;
    }

    - (void)cancel{
        [_task cancel];
    }

    /**
     *  处理网络请求的返回数据是否正确
     */
    - (void)handleResponseObject:(id)responseObject verifySuccess:(verifySuccess)verifySucdess andMessages:(messages)messages {
        BOOL isError = NO;
        NSString * message = nil;
        if (responseObject) {
            if (HHTRequestSuccess) {
                verifySucdess(YES);
            } else {
                if (self.dataList.count == 0) {
                    message = HHTServerMsg;
                }
            }
        } else {
            isError = YES;
        }
        messages(message, isError);
    }

    - (void)parse:(NSDictionary *)dictionary{
        //子类继承重写
    }

    @end
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容