MVVM+RAC总结

目录

一、什么是MVVM

二、为什么使用MVVM

三、RAC

3.1 RAC的用法

四、RAC与MVVM架构设计的优点

五、MVVM + RAC

5.1 MVVM使用指南

5.2 RAC坑


一、什么是MVVM

MVVM(Model - View/ViewController - ViewModel)是对MVC的一种变形设计模式,解决ViewController代码臃肿、View和Model模块耦合严重两个主要问题。抽出ViewModel来处理ViewController的业务逻辑,分离了UI代码和业务逻辑,并且ViewModel监听model事件,一旦发生变化更新视图,很好的解决了视图与模型的依赖性。看下图:

在MVVM 中,view 和 view controller正式联系在一起,我们把它们视为一个组件; view 和 view controller 都不能直接引用model,而是引用视图模型(viewModel);viewModel 是一个放置用户输入验证逻辑,视图显示逻辑,发起网络请求和其他代码的地方。

二、为什么使用MVVM

虽然MVC 依然是目前主流客户端编程框架,但同时它也被调侃成Massive View Controller(重量级视图控制器),想必大家在开发中无可避免被下面几个问题所困扰:

  • 厚重的ViewController
  • 遗失的网络逻辑
  • 较差的可测试性

为了避免和解决上述问题的产生,从MVC引申出来一种维护性较强、耦合性低的新的架构MVVM,主要目的是为了分离视图(View)和模型(Model)。

MVVM优势:

  • 低耦合:View 可以独立于Model变化和修改。
  • 可重用性:可以把一些视图逻辑放在一个 viewModel里面,让很多 view 重用这段视图逻辑。
  • 独立开发:开发人员可以专注于业务逻辑和数据的 viewModel开发。
  • 可测试:通常界面是比较难于测试的,而 MVVM 模式可以针对 viewModel来进行测试。
  • 兼容性:MVVM 可以兼容当下使用的MVC架构, 增加应用的可测试性。

三、RAC

ReactiveCocoa(简称为RAC),是由Github开源的一个应用于iOS和OS开发的新框架。结合了几种编程风格:函数式编程 和 响应式编程 ,使用RAC解决问题,就不需要考虑调用顺序,直接考虑结果,把每一次操作都写成一系列嵌套的方法中,使代码高聚合,方便管理。

函数式编程思想:是把操作尽量写成一系列嵌套的函数或者方法调用。

响应式编程思想:不需要考虑调用顺序,只需要考虑结果,产生一个事件,事件传播出去,然后影响结果。

3.1 RAC的用法

在RAC中最核心的类RACSignal:信号类,一般表示将来有数据传递,只要有数据改变,信号内部接收到数据,就会马上发出数据。

注意:

  • 信号类(RACSignal),只是表示当数据改变时,信号内部会发出数据,它本身不具备发送信号的能力,而是交给内部一个订阅者去发出。
  • 默认一个信号都是冷信号,也就是值改变了,也不会触发,只有订阅了这个信号,这个信号才会变为热信号,值改变了才会触发。
  • 如何订阅信号:调用信号RACSignal的subscribeNext就能订阅。

RACSignal使用步骤:

  • 创建信号 + (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe
  • 订阅信号. - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
  • 发送信号 - (void)sendNext:(id)value

案例:

//1.创建信号     
RACSignal *siganl = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {                                    

// 2.发送信号         
[subscriber sendNext:@1];                  

// 如果不在发送数据,最好发送信号完成,内部会自动调用[RACDisposable disposable]取消订阅信号。         
[subscriber sendCompleted];             

}];          

// 3.订阅信号  
[siganl subscribeNext:^(id x) {              
NSLog(@"接收到数据:%@",x);     
}];

2、RACSubject:信号提供者,自己可以充当信号,又能发送信号。

RACSubject使用步骤:

创建信号 [RACSubject subject],跟RACSiganl不一样,创建信号时没有block。

订阅信号 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock

发送信号 sendNext:(id)value

案例:

// 1.创建信号     
RACSubject *subject = [RACSubject subject];      

// 2.订阅信号     
[subject subscribeNext:^(id x) {         
NSLog(@"第一个订阅者%@",x);    
 }];         

// 3.发送信号     
[subject sendNext:@"1"];

3、RACReplaySubject:重复提供信号类,RACSubject的子类,如果一个信号每被订阅一次,就需要把之前的值重复发送一遍,使用重复提供信号类。

// 1.创建信号     
RACReplaySubject *replaySubject = [RACReplaySubject subject];      

// 2.发送信号    
 [replaySubject sendNext:@1];     

// 3.订阅信号    
 [replaySubject subscribeNext:^(id x) {         
 NSLog(@"第一个订阅者接收到的数据%@",x);     
}];

4、rac_signalForControlEvents:用于监听某个事件。

// 需求:把按钮点击事件转换为信号,点击按钮,就会发送信号 
@weakify(self)
[[[self.addMemberButton rac_signalForControlEvents:UIControlEventTouchUpInside] deliverOnMainThread] subscribeNext:^(id x) { 
@strongify(self)  
NSLog(@" %@ 按钮被点击了",self.addMemberButton); 
}];

5、rac_addObserverForName:用于监听某个通知。

//需求:截图事件日志上报
    [[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationUserDidTakeScreenshotNotification object:nil]
      takeUntil:[self rac_willDeallocSignal]] subscribeNext:^(id x) {
         [DXScreenshotHelper uploadLogInfo];

     }];

6、merge:将多个信号合并成一个信号。

- (RACSignal *)searchNeedsStopSignal

{

    if (!_searchNeedsStopSignal) {

        RACSignal *searchBarCancel = [self rac_signalForSelector:@selector(searchBarCancelButtonClicked:)

                                                    fromProtocol:@protocol(UISearchBarDelegate)];

        RACSignal *didSelectTableViewCell = [self rac_signalForSelector:@selector(tableView:didSelectRowAtIndexPath:)

                                                           fromProtocol:@protocol(UITableViewDelegate)];

        _searchNeedsStopSignal = [[RACSignal merge:@[searchBarCancel, didSelectTableViewCell]] takeUntil:[self rac_willDeallocSignal]];

    }

    return _searchNeedsStopSignal;

}

7、RAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于给某个对象的某个属性绑定。

RAC(self, unreadCount) = [[[[RACObserve([DXKFChannelManager shareInstance].chatChannel, totalUnreadCount) takeUntil:self.rac_willDeallocSignal] deliverOnMainThread] doNext:^(NSNumber *number) {

        DDLogVerbose(@"%s, Total unread count = %@", __FUNCTION__, number);

        uint64_t microAppID = [DXKFDataManager shareInstance].infoItem.microAppId;

        if (microAppID > 0) {

            [DXMicroAppLocalAlertManager addLocalAlertPointWithMicroAppId:microAppID count:number.integerValue];

        }

    }];

8、RACObserve(self, name):监听某个对象的某个属性,返回的是信号。

@weakify(self)

[[[RACObserve([UISDKManager sharedManager], connectStatus) takeUntil:self.rac_willDeallocSignal] deliverOnMainThread] subscribeNext:^(NSNumber *number) {

        @strongify(self);

        UIConnectStatus connectStatus = (UIConnectStatus)number.integerValue;

        switch (connectStatus) {

            case UIConnectStatusLogined:// 断网重连

            {

                [[DXKFDataManager shareInstance] cleanAllKFInfoMemoryCache];

                [[DXKFDataManager shareInstance] queryKFSystemEnabled:^(BOOL enabled) {

                    [self getKeFuCurrentStatus];

                    if (!enabled) {

                        [self resignNotice];

                    }

                }];

                break;

            }

            default:

                break;

        }

    }];

四、RAC与MVVM架构设计的优点

1、从上面RAC的用法可以看出,RAC的绑定,这是常用的。比如:用户信息展示界面->登录界面->登录成功->回到用户信息展示界面->展示用户信息,以往我们的做法通常是通知,也可以用协议、block什么的,一旦代码量多了过后,耦合度高,维护成本就会增加,而使用RAC的属性绑定、属性联合等一系列方法,将会有事半功倍的效果,这样就可以解决相当多的需求了。

2、MVVM搭建思路里面会涉及大量的属性绑定、事件传递来实现不同功能,运用RAC能大量简化代码,充分的降低了代码的耦合度,降低维护成本,思路更清晰。

五、MVVM + RAC

5.1 MVVM使用指南

viewController 尽量不涉及业务逻辑,让 viewModel 去做这些事情。

viewController 只是一个中间人,接收 view 的事件、调用 viewModel 的方法、响应 viewModel 的变化。

viewModel 绝对不能包含视图 view,不然就跟 view 产生了耦合,不方便复用和测试。

viewModel避免过于臃肿,否则重蹈Controller的覆辙,变得难以维护。

MVVM 配合一个绑定机制效果最好(PS:ReactiveCocoa)。

 

单人语音页面为例,通过下面的图解可以看出:

通过MVVM的架构模式,将所有的数据处理、业务逻辑处理等都抽离到了VM(ViewModel)里面,解放了Controller,降低了耦合性,利于重用和测试。

通过与RAC的结合使用,viewModel创建并发送信号,Controller订阅信号avatarImageSignal、currentRoutingChangeSignal、callQualitySignal、longTimeNoAnswerSignal等,而让界面实时进行更新,控制器单纯的进行UI操作,同时避免了通知代理等大量的使用,更易于维护,并且将点击事件通过RAC的方式,让所有代码一起处理,增加了可阅读性。

5.2 RAC坑

RAC/KVO forwardInvication: 方法互相覆盖问题

RAC循环调用问题/RAC内存泄漏问题

创建信号引起的循环引用情况

如下是几种简单的创建信号时引起的循环引用情况:

{
    // case 1 -- 这种情况较容易发现  
    self.someSignal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        NSLog(@"self is %@", self);
        [subscriber sendNext:self.someOtherProp];
        [subscriber sendCompleted];
        return nil;
    }];

    // case 2 -- 不太容易发现
    self.someSignal = [RACSignal return:self];

    // case 3 -- 不太容易发现
    self.someSignal = [[RACSignal return:@1] map:^id(id value) {
        return [NSString stringWithFormat:@"%@%@", self, value];
    }];
}

P.S: 我似乎忘记了,循环引用并不只是发生在异步block中

创建信号没有引起循环引用的情况

但如下这种情况没有循环引用:

- (RACSignal *)makeNewSignal
{
    return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        [subscriber sendNext:self.someProp];
        [subscriber sendNext:self.someOtherProp];
        [subscriber sendCompleted];
        return nil;
    }];
} 

引用情况如下:

这段代码存在风险,如果这个信号被某个全局变量持有了,`self`就永远不会被释放,这不属于循环引用,但是也是内存泄漏。内存泄漏不仅仅是循环引用,所有应该释放却没被释放的情况都算

订阅信号引起的循环引用情况

如下情况引发了循环引用:

- (void)retainCycleWhenSubscribe
{
    UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
    [[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
        self.someProperty = 42;
    }];
    self.button = button;
}

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: WPF是一种基于XAML语言的用户界面开发框架,使开发人员能够轻松创建漂亮和高效的桌面应用程序。MVVM是一种模式,它在WPF应用程序中广泛使用,它提供了一种分离UI和业务逻辑的方式,从而简化了代码结构。Prism是一个开源框架,它基于MVVM模式和WPF框架,提供了一些技术和工具,使开发人员能够更轻松地构建复杂的WPF应用程序。 使用WPF MVVM Prism框架可以帮助开发人员提高应用程序的可维护性和可扩展性。通过MVVM,开发人员可以创建一个适应各种应用程序场景的模型,并将其与UI分离。数据绑定和命令绑定使开发人员能够更轻松地将模型中的数据和逻辑与UI控件集成起来。Prism框架还提供了一些工具和功能,如模块化应用程序开发、事件聚合器、导航、对话框、异常处理等功能。这些工具使开发人员能够更轻松地构建复杂的应用程序,并提供了一种可扩展和可重用的方法。 总之,使用WPF MVVM Prism可以使开发人员更轻松地构建复杂的应用程序,并提高应用程序的可维护性和可扩展性。开发人员可以根据自己的需求选择使用这个框架来开发WPF应用程序,从而提高他们的工作效率和代码质量。 ### 回答2: WPF MVVM Prism是一种基于Windows Presentation Foundation(WPF)的软件开发模式,采用了MVVM(Model-View-ViewModel)设计模式和Prism框架来实现软件开发。MVVM是一种在WPF应用程序中使用的设计模式,它将应用程序的视图(View)与应用程序的模型(Model)分离开来,通过ViewModel来连接二者。Prism是一个帮助开发人员编写出复杂、可管控、可扩展和可维护的WPF应用程序的框架。 WPF MVVM Prism提供了很多好处:首先,它能实现软件的可重用性,通过将代码和视图分离,使得代码可以在多个不同的视图中重复使用,提高了开发效率。其次,它可以提高软件的可维护性,通过将逻辑和视图分离,确保逻辑代码与UI代码之间更少的耦合,提高了软件的可扩展性。此外,它还可以提高软件的可测试性,由于MVVM模式中将视图和逻辑分离开,所以开发人员可以更容易地编写出单元测试代码,来测试逻辑代码。最后,由于Prism框架提供了一些通用的,可定制的模块和服务,使得开发人员可以更快地实现常见功能、缩短开发时间。 总之,WPF MVVM Prism是一种高效、可扩展和易于维护的软件开发模式,它将视图和逻辑分离,提高了软件的可重用性和可测试性,同时也提高了软件的可扩展性和可复用性。 ### 回答3: WPF是一种桌面应用程序框架,它允许您为Windows创建高度交互式和可视化的应用程序界面。MVVM是一种软件编程模式,它将用户界面(UI)与业务逻辑分离,以便更好地实现可维护,可扩展和可组合的代码。Prism是一个WPF和Silverlight应用程序的组件库,它提供了一些可重用和可扩展的基础设施组件,帮助您更快速地构建、测试和维护WPF应用程序。因此,WPF MVVM Prism的组合可以帮助您更轻松高效地开发WPF应用程序。您可以使用MVVM模式来改进应用程序的结构和测试,使用Prism来更好地组织和可扩展您的代码,以及使用WPF来实现交互丰富的UI。同时还可以使用Prism提供的事件聚合器、导航器和模块化架构,实现更灵活的应用程序设计。WPF MVVM Prism的组合提供了一种更有效的方式来构建WPF应用程序,以满足现代用户体验的需求,并且更容易测试和维护,因此它已成为WPF应用程序开发的重要组合。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值