MVVM + RAC 模式详解

MVVM 是一种软件架构模式,它是 Martin FowlerPresentation Model 的一种变体,最先由微软的架构师 John Gossman 在 2005 年提出,并应用在微软的 WPFSilverlight 软件开发中。MVVM衍生于 MVC ,是对 MVC 的一种演进,它促进了 UI 代码与业务逻辑的分离。

说明:本文将采用理论与实践相结合的方式,重点介绍一个使用 MVVMRAC 开发的 iOS 开源项目 MVVMReactiveCocoa ,目的是希望能为你实践 MVVM 提供帮助。不过,在正式开始介绍正文之前,请你先思考以下三个问题:

  • MVCMVVM 有什么异同点,MVCMVVM 是怎样演进的;
  • RACMVVM 中扮演什么样的角色,MVVM 是否一定要结合 RAC 使用;
  • 如何将一个现有的 MVC 应用转变成一个 MVVM 应用,有哪些需要注意的地方。

带着以上问题,我们一起进入正文。

名词解释:本文中的 RACReactiveCocoa 的缩写。

MVC

MVCiOS 开发中使用最普遍的架构模式,同时也是苹果官方推荐的架构模式。MVC 代表的是 Model–view–controller ,它们之间的关系如下:

image

是的,MVC 看上去棒极了,model 代表数据,view 代表 UI ,而 controller 则负责协调它们两者之间的关系。然而,尽管从技术上看 viewcontroller 是相互独立的,但事实上它们几乎总是结对出现,一个 view 只能与一个 controller 进行匹配,反之亦然。既然如此,那我们为何不将它们看作一个整体呢:

image

因此,M-VC 可能是对 iOS 中的 MVC 模式更为准确的解读。在一个典型的 MVC 应用中,controller 由于承载了过多的逻辑,往往会变得臃肿不堪,所以 MVC 也经常被人调侃成 Massive View Controller

iOS architecture, where MVC stands for Massive View Controller.

坦白说,有一部分逻辑确实是属于 controller 的,但是也有一部分逻辑是不应该被放置在 controller 中的。比如,将 model 中的 NSDate 转换成 view 可以展示的 NSString 等。在 MVVM 中,我们将这些逻辑统称为展示逻辑。

MVVM

因此,一种可以很好地解决 Massive View Controller 问题的办法就是将 controller 中的展示逻辑抽取出来,放置到一个专门的地方,而这个地方就是 viewModel 。其实,我们只要在上图中的 M-VC 之间放入 VM ,就可以得到 MVVM 模式的结构图:

image

从上图中,我们可以非常清楚地看到 MVVM 中四个组件之间的关系。:除了 viewviewModelmodel 之外,MVVM 中还有一个非常重要的隐含组件 binder

  • view :由 MVC 中的 viewcontroller 组成,负责 UI 的展示,绑定 viewModel 中的属性,触发 viewModel 中的命令;
  • viewModel :从 MVCcontroller 中抽取出来的展示逻辑,负责从 model 中获取 view 所需的数据,转换成 view 可以展示的数据,并暴露公开的属性和命令供 view 进行绑定;
  • model :与 MVC 中的 model 一致,包括数据模型、访问数据库的操作和网络请求等;
  • binder :在 MVVM 中,声明式的数据和命令绑定是一个隐含的约定,它可以让开发者非常方便地实现 viewviewModel 的同步,避免编写大量繁杂的样板化代码。在微软的 MVVM 实现中,使用的是一种被称为 XAML 的标记语言。

ReactiveCocoa

尽管,在 iOS 开发中,系统并没有提供类似的框架可以让我们方便地实现 binder 功能,不过,值得庆幸的是,GitHub 开源的 RAC ,给了我们一个非常不错的选择。

RAC 是一个 iOS 中的函数式响应式编程框架,它受 Functional Reactive Programming 的启发,是 Justin Spahr-SummersJosh Abernathy 在开发 GitHub for Mac 过程中的一个副产品,它提供了一系列用来组合和转换值流的 API 。如需了解更多关于 RAC 的信息,可以阅读我的上一篇文章《ReactiveCocoa v2.5 源码解析之架构总览》

iOSMVVM 实现中,我们可以使用 RAC 来在 viewviewModel 之间充当 binder 的角色,优雅地实现两者之间的同步。此外,我们还可以把 RAC 用在 model 层,使用 Signal 来代表异步的数据获取操作,比如读取文件、访问数据库和网络请求等。说明RAC 的后一个应用场景是与 MVVM 无关的,也就是说,我们同样可以在 MVCmodel 层这么用。

小结

综上所述,我们只要将 MVC 中的 controller 中的展示逻辑抽取出来,放置到 viewModel 中,然后通过一定的技术手段,比如 RAC 来同步 viewviewModel ,就完成了 MVCMVVM 的转变。

Talk is cheap. Show me the code.

下面,我们直接上代码,一起来看一个 MVC 模式转换成 MVVM 模式的示例。首先是 model 层的代码 Person

@interface Person : NSObject

- (instancetype)initwithSalutation:(NSString *)salutation firstName:(NSString *)firstName lastName:(NSString *)lastName birthdate:(NSDate *)birthdate;

@property (nonatomic, copy, readonly) NSString *salutation;
@property (nonatomic, copy, readonly) NSString *firstName;
@property (nonatomic, copy, readonly) NSString *lastName;
@property (nonatomic, copy, readonly) NSDate *birthdate;

@end

然后是 view 层的代码 PersonViewController ,在 viewDidLoad 方法中,我们将 Person 中的属性进行一定的转换后,赋值给相应的 view 进行展示:

- (void)viewDidLoad {
    [super viewDidLoad];

    if (self.model.salutation.length > 0) {
        self.nameLabel.text = [NSString stringWithFormat:@"%@ %@ %@", self.model.salutation, self.model.firstName, self.model.lastName];
    } else {
        self.nameLabel.text = [NSString stringWithFormat:@"%@ %@", self.model.firstName, self.model.lastName];
    }

    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    [dateFormatter setDateFormat:@"EEEE MMMM d, yyyy"];
    self.birthdateLabel.text = [dateFormatter stringFromDate:model.birthdate];
}

接下来,我们引入一个 viewModel ,将 PersonViewController 中的展示逻辑抽取到这个 PersonViewModel 中:

@interface PersonViewModel : NSObject

- (instancetype)initWithPerson:(Person *)person;

@property (nonatomic, strong, readonly) Person *person;
@property (nonatomic, copy, readonly) NSString *nameText;
@property (nonatomic, copy, readonly) NSString *birthdateText;

@end

@implementation PersonViewModel

- (instancetype)initWithPerson:(Person *)person {
    self = [super init];
    if (self) {
       _person = person;

      if (person.salutation.length > 0) {
          _nameText = [NSString stringWithFormat:@"%@ %@ %@", self.person.salutation, self.person.firstName, self.person.lastName];
      } else {
          _nameText = [NSString stringWithFormat:@"%@ %@", self.person.firstName, self.person.lastName];
      }

      NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
      [dateFormatter setDateFormat:@"EEEE MMMM d, yyyy"];
      _birthdateText = [dateFormatter stringFromDate:person.birthdate];
    }
    return self;
}

@end

最终,PersonViewController 将会变得非常轻量级:

- (void)viewDidLoad {
    [super viewDidLoad];

    self.nameLabel.text = self.viewModel.nameText;
    self.birthdateLabel.text = self.viewModel.birthdateText;
}

怎么样?其实 MVVM 并没有想像中的那么难吧,而且更重要的是它也没有破坏 MVC 的现有结构,只不过是移动了一些代码,仅此而已。好了,说了这么多,那 MVVM 相比 MVC 到底有哪些好处呢?我想,主要可以归纳为以下三点:

  • 由于展示逻辑被抽取到了 viewModel 中,所以 view 中的代码将会变得非常轻量级;
  • 由于 viewModel 中的代码是与 UI 无关的,所以它具有良好的可测试性;
  • 对于一个封装了大量业务逻辑的 model 来说,改变它可能会比较困难,并且存在一定的风险。在这种场景下,viewModel 可以作为 model 的适配器使用,从而避免对 model 进行较大的改动。

通过前面的示例,我们对第一点已经有了一定的感触;至于第三点,可能对于一个复杂的大型应用来说,才会比较明显;下面,我们还是使用前面的示例,来直观地感受下第二点好处:

SpecBegin(Person)
    NSString *salutation = @"Dr.";
    NSString *firstName = @"first";
    NSString *lastName = @"last";
    NSDate *birthdate = [NSDate dateWithTimeIntervalSince1970:0];

    it (@"should use the salutation available. ", ^{
        Person *person = [[Person alloc] initWithSalutation:salutation firstName:firstName lastName:lastName birthdate:birthdate];
        PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person];
        expect(viewModel.nameText).to.equal(@"Dr. first last");
    });

    it (@"should not use an unavailable salutation. ", ^{
        Person *person = [[Person alloc] initWithSalutation:nil firstName:firstName lastName:lastName birthdate:birthdate];
        PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person];
        expect(viewModel.nameText).to.equal(@"first last");
    });

    it (@"should use the correct date format. ", ^{
        Person *person = [[Person alloc] initWithSalutation:nil firstName:firstName lastName:lastName birthdate:birthdate];
        PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person];
        expect(viewModel.birthdateText).to.equal(@"Thursday January 1, 1970");
    });
SpecEnd

对于 MVVM 来说,我们可以把 view 看作是 viewModel 的可视化形式,viewModel 提供了 view所需的数据和命令。因此,viewModel 的可测试性可以帮助我们极大地提高应用的质量。

MVVMReactiveCocoa

接下来,我们进入本文的第二部分,重点介绍一个使用 MVVMRAC 开发的开源项目 MVVMReactiveCocoa说明,本文将主要介绍这个应用的架构和设计思路,希望可以为你实践 MVVM 提供一个真实的参考案例,有些架构并非是 MVVM 所必须的,而是我们为了更顺畅地使用 MVVM 而引入的,特别是 ViewModel-Based Navigation 。所以,请你在实践的过程中能够结合自身应用的实际情况做出相应的取舍,灵活处理。最后,我们将以登录界面为例,一起探讨下 MVVM的实践思路。

说明,以下内容均基于 MVVMReactiveCocoav2.1.1 标签进行展开,并且对部分无关代码做了删减。

类图

为了方便我们从宏观上了解 MVVMReactiveCocoa 的整体结构,我们先来看看它的类图:

MVVMReactiveCocoa-v2.1.1

从上图中,我们可以看到,在 MVVMReactiveCocoa 中主要有两大继承体系:

  • 用蓝色标识出来的 viewModel 的继承体系,基类为 MRCViewModel
  • 用红色标识出来的 view 的继承体系,基类为 MRCViewController

除了提供与系统基类 UIViewController 相对应的基类 MRCViewModel/MRCViewController 外,还提供了与系统基类 UITableViewControllerUITabBarController 相对应的基类 MRCTableViewModel/MRCTableViewControllerMRCTabBarViewModel/MRCTabBarController ,其中基类 MRCTableViewModel/MRCTableViewController 的使用最为普遍。

说明,之所以通过基类的方式来组织 MVVMReactiveCocoa ,一方面是因为主要开发者只有我一个人,这个方案非常容易实施;另一方面是因为通过基类的方式可以尽可能简单地实现代码重用,提高开发效率。

服务总线

经过前面的探讨,我们已经知道了 MVVM 中的 viewModel 的主要职责就是从 model 层获取 view所需的数据,并且将这些数据转换成 view 能够展示的形式。因此,为了方便 viewModel 层调用 model 层中的所有服务,并且统一管理这些服务的创建,我使用抽象工厂模式将 model 层的所有服务集中管理了起来,结构图如下:

image

从上图中,我们可以看出,在服务总线类 MRCViewModelServices/MRCViewModelServicesImpl 中,主要包括以下三个方面的内容:

  • 应用自有的服务类,用柚黄色进行了标识,包括 MRCAppStoreService/MRCAppStoreServiceImplMRCRepositoryService/MRCRepositoryServiceImpl 两个服务类;
  • 第三方 GitHub 提供的 API 框架,用天蓝色进行了标识,主要包括 OCTClient 服务类;
  • 应用的导航服务,用藻绿色进行了标识,包括 MRCNavigationProtocol 协议和实现类 MRCViewModelServicesImpl 等。

其中,前两者都是以信号的形式对 viewModel 层提供服务,代表异步的网络请求等数据获取操作,而我们在 viewModel 层则可以通过订阅信号的形式获取到所需的数据。此外,服务总线还实现了 MRCNavigationProtocol 协议,它的内容如下:

@protocol MRCNavigationProtocol <NSObject>

- (void)pushViewModel:(MRCViewModel *)viewModel animated:(BOOL)animated;

- (void)popViewModelAnimated:(BOOL)animated;

- (void)popToRootViewModelAnimated:(BOOL)animated;

- (void)presentViewModel:(MRCViewModel *)viewModel animated:(BOOL)animated completion:(VoidBlock)completion;

- (void)dismissViewModelAnimated:(BOOL)animated completion:(VoidBlock)completion;

- (void)resetRootViewModel:(MRCViewModel *)viewModel;

@end

看上去是不是有点眼熟?是的,MRCNavigationProtocol 协议其实就是参照系统的导航操作定义出来的,用来实现 ViewModel-Based 的导航服务。注意,服务总线类 MRCViewModelServicesImpl 其实并没有真正实现 MRCNavigationProtocol 协议中声明的操作,只不过是实现了一些空操作而已:

- (void)pushViewModel:(MRCViewModel *)viewModel animated:(BOOL)animated {}

- (void)popViewModelAnimated:(BOOL)animated {}

- (void)popToRootViewModelAnimated:(BOOL)animated {}

- (void)presentViewModel:(MRCViewModel *)viewModel animated:(BOOL)animated completion:(VoidBlock)completion {}

- (void)dismissViewModelAnimated:(BOOL)animated completion:(VoidBlock)completion {}

- (void)resetRootViewModel:(MRCViewModel *)viewModel {}

那么,我们是怎么实现 ViewModel-Based 的导航操作的呢?用 MRCViewModelServicesImpl 来实现这些空操作到底有什么用意?为什么要这么做,目的是为了什么?兄台,莫急,请接着看下一小节的内容。

ViewModel-Based Navigation

我们先来思考一个问题,就是我们为什么要实现 ViewModel-Based 的导航操作呢?直接在 view 层使用系统的 push/present 等操作来完成导航不就好了么?我总结了一下这么做的理由,主要有以下三点:

  • 从理论上来说,MVVM 模式的应用应该是以 viewModel 为驱动来运转的;
  • 根据我们前面对 MVVM 的探讨,viewModel 提供了 view 所需的数据和命令。因此,我们往往可以直接在命令执行成功后使用 doNext 顺带就把导航操作给做了,一气呵成;
  • 这样可以使 view 更加轻量级,只需要绑定 viewModel 提供的数据和命令即可。

既然如此,那我们究竟要如何实现 ViewModel-Based 的导航操作呢?我们都知道 iOS 中的导航操作无外乎两种,push/poppresent/dismiss ,前者是 UINavigationController 特有的功能,而后者是所有 UIViewController 都具备的功能。注意UINavigationController 也是 UIViewController 的子类,所以它也同样具备 present/dismiss 的功能。因此,从本质上来说,不管我们要实现什么样的导航操作,最终都是离不开 push/poppresent/dismiss 的。

目前,MVVMReactiveCocoa 的做法是在 view 层维护一个 NavigationController 的堆栈 MRCNavigationControllerStack ,不管是 push/pop 还是 present/dismiss ,都使用栈顶的 NavigationController 来执行导航操作,并且保证 present 出来的是一个 NavigationController

接下来,我们一起来看看 MVVMReactiveCocoa 在执行了 push/poppresent/dismiss 操作后视图层次结构的变化过程。首先,我们来看看用户在登录成功后进入到首页时应用的视图层次结构图:

image

此时,应用展示的界面是 NewsViewController 。在 MRCNavigationControllerStack 堆栈中只有 NavigationController0 一个元素;而 NavigationController1 并没有在 MRCNavigationControllerStack 堆栈中,这是因为需要支持 TabBarController 的滑动切换而设计的视图层次结构,是首页比较特殊的一个地方。更多信息可以查看 GitHub 开源库 WXTabBarController ,在这里,我们不用太过于关心这个问题,只需要理解原理就好了。

接下来,当用户在 NewsViewController 界面,点击了某一个 cell ,通过 push 的方式,进入到仓库详情界面时,应用的视图层次结构图如下:

image

应用通过 MRCNavigationControllerStack 栈顶的元素 NavigationController0 ,将仓库详情界面 push 到了自身的堆栈中。此时,应用展示的界面是被 push 进来的仓库详情界面 RepoDetailViewController 。最后,当用户在仓库详情界面,点击左下角的切换分支按钮,通过 present 的方式,弹出分支选择界面时,应用的视图层次结构图如下:

image

应用通过 MRCNavigationControllerStack 栈顶的元素 NavigationController0 ,将 NavigationController5present 的方式弹出来。此时,应用展示的是 NavigationController5 的根视图 SelectBranchOrTagViewController说明,由于 popdismisspushpresent 互为逆操作,所以只要按照从下到上的顺序看上面的视图层次结构图即可,这里不再赘述。

等等,如果我没有记错的话,MRCNavigationControllerStack 堆栈是在 view 层,而服务总线类 MRCViewModelServicesImpl 是在 viewModel 层的。据我所知,viewModel 层是不能引入 view 层的任何东西的,更严格的说,是不能引入任何 UIKit 中的东西的,否则就违背了 MVVM 的基本原则,并且也会散失 viewModel 的可测试性。在这个前提下,你要如何让这两者产生关联呢?

没错,这就是 MRCViewModelServicesImpl 中之所以实现那些空操作的目的所在了。viewModel 通过调用 MRCViewModelServicesImpl 中的空操作来表明需要执行相应的导航操作,而 MRCNavigationControllerStack 则通过 Hook 来捕获这些空操作,然后使用栈顶的 NavigationController 来执行真正的导航操作:

- (void)registerNavigationHooks {
    @weakify(self)
    [[(NSObject *)self.services
        rac_signalForSelector:@selector(pushViewModel:animated:)]
        subscribeNext:^(RACTuple *tuple) {
            @strongify(self)
            UIViewController *viewController = (UIViewController *)[MRCRouter.sharedInstance viewControllerForViewModel:tuple.first];
            [self.navigationControllers.lastObject pushViewController:viewController animated:[tuple.second boolValue]];
        }];

    [[(NSObject *)self.services
        rac_signalForSelector:@selector(popViewModelAnimated:)]
        subscribeNext:^(RACTuple *tuple) {
          @strongify(self)
            [self.navigationControllers.lastObject popViewControllerAnimated:[tuple.first boolValue]];
        }];

    [[(NSObject *)self.services
        rac_signalForSelector:@selector(popToRootViewModelAnimated:)]
        subscribeNext:^(RACTuple *tuple) {
            @strongify(self)
            [self.navigationControllers.lastObject popToRootViewControllerAnimated:[tuple.first boolValue]];
        }];

    [[(NSObject *)self.services
        rac_signalForSelector:@selector(presentViewModel:animated:completion:)]
        subscribeNext:^(RACTuple *tuple) {
          @strongify(self)
            UIViewController *viewController = (UIViewController *)[MRCRouter.sharedInstance viewControllerForViewModel:tuple.first];

            UINavigationController *presentingViewController = self.navigationControllers.lastObject;
            if (![viewController isKindOfClass:UINavigationController.class]) {
                viewController = [[MRCNavigationController alloc] initWithRootViewController:viewController];
            }
            [self pushNavigationController:(UINavigationController *)viewController];

            [presentingViewController presentViewController:viewController animated:[tuple.second boolValue] completion:tuple.third];
        }];

    [[(NSObject *)self.services
        rac_signalForSelector:@selector(dismissViewModelAnimated:completion:)]
        subscribeNext:^(RACTuple *tuple) {
            @strongify(self)
            [self popNavigationController];
            [self.navigationControllers.lastObject dismissViewControllerAnimated:[tuple.first boolValue] completion:tuple.second];
        }];

    [[(NSObject *)self.services
        rac_signalForSelector:@selector(resetRootViewModel:)]
        subscribeNext:^(RACTuple *tuple) {
            @strongify(self)
            [self.navigationControllers removeAllObjects];

            UIViewController *viewController = (UIViewController *)[MRCRouter.sharedInstance viewControllerForViewModel:tuple.first];

            if (![viewController isKindOfClass:[UINavigationController class]]) {
                viewController = [[MRCNavigationController alloc] initWithRootViewController:viewController];
                ((UINavigationController *)viewController).delegate = self;
                [self pushNavigationController:(UINavigationController *)viewController];
            }

            MRCSharedAppDelegate.window.rootViewController = viewController;
        }];
}

通过 Hook 的方式,我们最终实现了 ViewModel-Based 的导航操作,并且在 viewModel 层中也没有引入 view 层的任意东西,实现了解耦合。

Router

还有一点值得一提的是,我们在 viewModel 中调用导航操作的时候,只传入了 viewModel 的实例作为参数,那么我们在 MRCNavigationControllerStack 中执行真正的导航操作时,怎么才能知道要跳转到哪个界面呢?为此,我们配置了一个从 viewModelview 的映射,并且约定了一个统一的初始化 view 的方法 initWithViewModel:

- (MRCViewController *)viewControllerForViewModel:(MRCViewModel *)viewModel {
    NSString *viewController = self.viewModelViewMappings[NSStringFromClass(viewModel.class)];

    NSParameterAssert([NSClassFromString(viewController) isSubclassOfClass:[MRCViewController class]]);
    NSParameterAssert([NSClassFromString(viewController) instancesRespondToSelector:@selector(initWithViewModel:)]);

    return [[NSClassFromString(viewController) alloc] initWithViewModel:viewModel];
}

- (NSDictionary *)viewModelViewMappings {
    return @{
      @"MRCLoginViewModel": @"MRCLoginViewController",
        @"MRCHomepageViewModel": @"MRCHomepageViewController",
        @"MRCRepoDetailViewModel": @"MRCRepoDetailViewController",
        ...
    };
}

登录界面

最后,我们一起来看看登录界面中 viewModelview 的部分关键代码,探讨一下 MVVM 的具体实践过程。说明,我们将会尽可能地回避具体的业务逻辑,重点关注 MVVM 的实践思路。下面是登录界面的截图:

image

其中,主要的界面元素有:

  • 一个用于展示用户头像的按钮 avatarButton
  • 用于输入账号和密码的输入框 usernameTextFieldpasswordTextField
  • 一个直接登录的按钮 loginButton 和一个跳转到浏览器授权登录的按钮 browserLoginButton

分析:根据我们前面对 MVVM 的探讨,viewModel 需要提供 view 所需的数据和命令。因此,MRCLoginViewModel.h 头文件的内容大致如下:

@interface MRCLoginViewModel : MRCViewModel

@property (nonatomic, copy, readonly) NSURL *avatarURL;
@property (nonatomic, copy) NSString *username;
@property (nonatomic, copy) NSString *password;

@property (nonatomic, strong, readonly) RACSignal *validLoginSignal;
@property (nonatomic, strong, readonly) RACCommand *loginCommand;
@property (nonatomic, strong, readonly) RACCommand *browserLoginCommand;

@end

非常直观,其中需要特别说明的是 validLoginSignal 属性代表的是登录按钮是否可用,它将会与 view 中登录按钮的 enabled 属性进行绑定。接着,我们来看看 MRCLoginViewModel.m 的实现文件中的部分关键代码:

@implementation MRCLoginViewModel

- (void)initialize {
    [super initialize];

    RAC(self, avatarURL) = [[RACObserve(self, username)
        map:^(NSString *username) {
            return [[OCTUser mrc_fetchUserWithRawLogin:username] avatarURL];
        }]
        distinctUntilChanged];

    self.validLoginSignal = [[RACSignal
      combineLatest:@[ RACObserve(self, username), RACObserve(self, password) ]
        reduce:^(NSString *username, NSString *password) {
          return @(username.length > 0 && password.length > 0);
        }]
        distinctUntilChanged];

    @weakify(self)
    void (^doNext)(OCTClient *) = ^(OCTClient *authenticatedClient) {
        @strongify(self)
        MRCHomepageViewModel *viewModel = [[MRCHomepageViewModel alloc] initWithServices:self.services params:nil];
        dispatch_async(dispatch_get_main_queue(), ^{
            [self.services resetRootViewModel:viewModel];
        });
    };

    [OCTClient setClientID:MRC_CLIENT_ID clientSecret:MRC_CLIENT_SECRET];

    self.loginCommand = [[RACCommand alloc] initWithSignalBlock:^(NSString *oneTimePassword) {
      @strongify(self)
        OCTUser *user = [OCTUser userWithRawLogin:self.username server:OCTServer.dotComServer];
        return [[OCTClient
          signInAsUser:user password:self.password oneTimePassword:oneTimePassword scopes:OCTClientAuthorizationScopesUser | OCTClientAuthorizationScopesRepository note:nil noteURL:nil fingerprint:nil]
            doNext:doNext];
    }];

    self.browserLoginCommand = [[RACCommand alloc] initWithSignalBlock:^(id input) {
        return [[OCTClient
          signInToServerUsingWebBrowser:OCTServer.dotComServer scopes:OCTClientAuthorizationScopesUser | OCTClientAuthorizationScopesRepository]
            doNext:doNext];
    }];
}

@end

  • 当用户输入的用户名发生变化时,调用 model 层的方法查询本地数据库中缓存的用户数据,并返回 avatarURL 属性;
  • 当用户输入的用户名或密码发生变化时,判断用户名和密码的长度是否均大于 0 ,如果是则登录按钮可用,否则不可用;
  • loginCommandbrowserLoginCommand 命令执行成功时,调用 doNext 代码块,使用服务总线中的方法 resetRootViewModel: 进入首页。

接下来,我们来看看 MRCLoginViewController 中的部分关键代码:

@implementation MRCLoginViewController

- (void)bindViewModel {
    [super bindViewModel];

    @weakify(self)
    [RACObserve(self.viewModel, avatarURL) subscribeNext:^(NSURL *avatarURL) {
      @strongify(self)
        [self.avatarButton sd_setImageWithURL:avatarURL forState:UIControlStateNormal placeholderImage:[UIImage imageNamed:@"default-avatar"]];
    }];

    RAC(self.viewModel, username)  = self.usernameTextField.rac_textSignal;
    RAC(self.viewModel, password)  = self.passwordTextField.rac_textSignal;
    RAC(self.loginButton, enabled) = self.viewModel.validLoginSignal;

    [[self.loginButton
        rac_signalForControlEvents:UIControlEventTouchUpInside]
        subscribeNext:^(id x) {
            @strongify(self)
            [self.viewModel.loginCommand execute:nil];
        }];

    [[self.browserLoginButton
        rac_signalForControlEvents:UIControlEventTouchUpInside]
        subscribeNext:^(id x) {
            @strongify(self)
            NSString *message = [NSString stringWithFormat:@"“%@” wants to open “Safari”", MRC_APP_NAME];

            UIAlertController *alertController = [UIAlertController alertControllerWithTitle:nil
                                                                                     message:message
                                                                              preferredStyle:UIAlertControllerStyleAlert];

            [alertController addAction:[UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleCancel handler:NULL]];
            [alertController addAction:[UIAlertAction actionWithTitle:@"Open" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
                @strongify(self)
                [self.viewModel.browserLoginCommand execute:nil];
            }]];

            [self presentViewController:alertController animated:YES completion:NULL];
        }];
}

@end

  • 观察 viewModelavatarURL 属性的变化,然后设置 avatarButton 中的图片;
  • viewModel 中的 usernamepassword 属性分别与 usernameTextFieldpasswordTextField 输入框中的内容进行绑定;
  • loginButtonenabled 属性与 viewModelvalidLoginSignal 属性进行绑定;
  • loginButtonbrowserLoginButton 按钮被点击时分别执行 loginCommandbrowserLoginCommand 命令。

综上所述,我们将 MRCLoginViewController 中的展示逻辑抽取到 MRCLoginViewModel 中后,使得 MRCLoginViewController 中的代码更加简洁和清晰。实践 MVVM 的关键点在于,我们要能够分析清楚 viewModel 需要暴露给 view 的数据和命令,这些数据和命令能够代表 view 当前的状态。

总结

首先,我们从理论出发介绍了 MVCMVVM 各自的概念以及从 MVCMVVM 的演进过程;接着,介绍了 RACMVVM 中的两个使用场景;最后,我们从实践的角度,重点介绍了一个使用 MVVMRAC 开发的开源项目 MVVMReactiveCocoa 。总的来说,我认为 iOS 中的 MVVM 可以分为以下三种不同的实践程度,它们分别对应不同的适用场景:

  • MVVM + KVO ,适用于现有的 MVC 项目,想转换成 MVVM 但是不打算引入 RAC 作为 binder 的团队;
  • MVVM + RAC ,适用于现有的 MVC 项目,想转换成 MVVM 并且打算引入 RAC 作为 binder 的团队;
  • MVVM + RAC + ViewModel-Based Navigation ,适用于全新的项目,想实践 MVVM 并且打算引入 RAC 作为 binder ,然后也想实践 ViewModel-Based Navigation 的团队。

写在最后,希望这篇文章能够打消你对 MVVM 模式的顾虑,赶快行动起来吧。

原文链接:http://blog.leichunfeng.com/blog/2016/02/27/mvvm-with-reactivecocoa/

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

推荐阅读更多精彩内容