转载

iOS开发模式MVVM 2分离业务逻辑

我想大多初始化定义的代码都能看懂,也没有叙述的必要,我主要把我觉得最主要的几个部分叙述一下。

代理 Protocol

为什么先说代理。这也是之前和慎哥也说过cell里面有点击事件怎么写。(不知道这么写对不对...后面看一部分源码之后会再聊这个话题,先留坑)代理可以干什么,跨Controller传值。跨Controller调方法。

我们之前写代理,都是在某个Controller里的.h文件最上面创建代理。但实际上,Xcode提供了专门的Protocol文件。之前没有搞明白也是因为我觉得代理必须要再某个文件里声明。

所以,当我们单独声明了一个Protocol文件的时候,就意味着,这个Protocol可以像一个类一样来声明变量了。

RACCommand

这个东西是做啥的。我只能说我解释不好,因为我没有完全理解这个东西。我只能把我理解的说出来。

我们的按钮都有一个点击事件。当点击的时候出发一个函数。

但是,当我们用了RACCommand的时候,我们的点击事件就可以这么写了。

self.pushBtn.rac_command = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {     NSLog(@"test signal");     return [RACSignal empty]; }];

和我之前写过的这种写法又不一样了。这种写法其实是对button addtarget那个方法的一个rac式的封装。

[[self.testBtn rac_signalForControlEvents:UIControlEventTouchUpInside]                             subscribeNext:^(id x) {     TestViewController *testVC = [[TestViewController alloc] init];     [self.navigationController pushViewController:testVC animated:YES]; }];

用第一种方法的原因呢,我们可以把点击事件赋值成为一个RACCommand类型的属性。既然可以变成一个属性了,那么就可以重新定义并且赋值。

分离点击事件

我们新建一个FirstViewModel,在FirstViewModel.h中增加一个

@property (strong, nonatomic) RACCommand *excutePush;

然后我们让刚刚在FirstViewController里定义的那个pushBtn的rac_command,将它赋值为FirstViewModel里的excutePush

self.pushBtn.rac_command = self.viewModel.excutePush;

这么一赋值,就达成了一个目的。按钮的事件在viewModel中执行。

需要注意的是,RACCommand的block返回的时一个signal。所以,在viewModel中,我们要这么写

    self.excutePush = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {         NSLog(@"test signal");         return [RACSignal empty];     }];

这样,我们就实现了对点击事件的分离。

分离跳转

分离跳转,就要用到我开始说的代理了。我们做一个MVVMdemoService的代理,我放在Protocol文件夹里了。这个代理就声明了一个方法

- (void)pushViewModel:(id)viewModel;

我们在NSLog(@"test signal");下面增加

[self.service pushViewModel:viewModel];

我们还要实现这个代理。

在ray中的 教程 和雷大神的 MVVMReactiveCocoa 都是新建了一个Impl文件,专门用来跳转,基本思路就是navigation本身就是一个堆栈,其实所有的vc都是在navigation其中的。我们只要控制这个这个最基本的navigation的跳转就可以了。

我们新建一个MVVMdemoImpl文件,里面有一个初始化的方法。

- (instancetype)initWithNavigationController:(UINavigationController *)navigationController

我们把根navigation传进去来进行逻辑跳转。

我们在初始化的时候加上这两句核心代码。

self.demoImpl = [[MVVMdemoImpl alloc] initWithNavigationController:self.naviVC]; self.firstViewModel = [[FirstViewModel alloc] initWithService:self.demoImpl]; 

第一句话是声明了一个demoImpl,将navigation穿进去。第二句话是声明了一个firstViewModel,把第一个demoImpl传了进去。这个地方的疑问在于,我们声明的时候是

-(instancetype)initWithService:(id<MVVMdemoService>)service

参数是一个代理。我们传的时一个NSObject类型的值。而且在MVVMdemoImpl.m中,也没有对于代理的赋值为自己。

我是这么理解的,如果有不对,还请谅解。

当我们加入代理MVVMdemoService在MVVMdemoImpl的时候,因为MVVMdemoImpl是一个nsobject类型,而MVVMdemoService也是一个nsobject类型。所以,此时我们的MVVMdemoImpl具有MVVMdemoService的属性了,我们做的就是,让FirstViewModel里的service的代理即是MVVMdemoImpl。

简单的说。

就是MVVMdemoImpl == FirstViewModel.service

那么这样的话,我们的service就可以执行MVVMdemoImpl里的pushViewModel方法了。

小结

其实上面的最后AppDelegate的赋值代理这一块还不是很明白。有时间再说说吧。好久没写博客就想发一篇了。

以上完整代码在 MVVMdemo

参考链接

1. leichunfeng/MVVMReactiveCocoa

2. ReactiveCocoa Essentials: Understanding and Using RACCommand

3. Why Does RACCommand's block return a signal?

4. MVVM Tutorial with ReactiveCocoa: Part 1/2

正文到此结束
Loading...