{"id":21654502,"url":"https://github.com/wujunyang/mvvmreactivecocoademo","last_synced_at":"2025-05-08T21:35:20.030Z","repository":{"id":94338158,"uuid":"49184190","full_name":"wujunyang/MVVMReactiveCocoaDemo","owner":"wujunyang","description":"ReactiveCocoa的知识点及MVVM模式运用（不断更新中....）","archived":false,"fork":false,"pushed_at":"2017-08-14T12:35:32.000Z","size":4040,"stargazers_count":263,"open_issues_count":2,"forks_count":65,"subscribers_count":14,"default_branch":"master","last_synced_at":"2025-01-26T04:05:38.192Z","etag":null,"topics":["mvvm","mvvmdemo","objective-c","reactivecocoa","reactivecocoa-mvvm","unittests"],"latest_commit_sha":null,"homepage":"","language":"Objective-C","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":null,"status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/wujunyang.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":null,"funding":null,"license":null,"code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null}},"created_at":"2016-01-07T05:51:49.000Z","updated_at":"2025-01-13T09:38:25.000Z","dependencies_parsed_at":"2023-04-07T11:25:25.094Z","dependency_job_id":null,"html_url":"https://github.com/wujunyang/MVVMReactiveCocoaDemo","commit_stats":null,"previous_names":[],"tags_count":0,"template":false,"template_full_name":null,"repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/wujunyang%2FMVVMReactiveCocoaDemo","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/wujunyang%2FMVVMReactiveCocoaDemo/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/wujunyang%2FMVVMReactiveCocoaDemo/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/wujunyang%2FMVVMReactiveCocoaDemo/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/wujunyang","download_url":"https://codeload.github.com/wujunyang/MVVMReactiveCocoaDemo/tar.gz/refs/heads/master","host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":238044095,"owners_count":19407128,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2022-07-04T15:15:14.044Z","host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["mvvm","mvvmdemo","objective-c","reactivecocoa","reactivecocoa-mvvm","unittests"],"created_at":"2024-11-25T08:28:04.334Z","updated_at":"2025-02-10T02:10:53.752Z","avatar_url":"https://github.com/wujunyang.png","language":"Objective-C","funding_links":[],"categories":[],"sub_categories":[],"readme":"\n# MVVMReactiveCocoaDemo介绍\n\nMVVMReactiveCocoaDemo是一个以学习ReactiveCocoa为主的项目，里面包含关于ReactiveCocoa基础知识点及如何结合MVVM进行开发，还有部分关于单元测试的知识，可以快速了解关于ReactiveCocoa如何运用在项目中，项目中的实例都有相应的介绍跟输出说明；项目中还有几个关于MVVM的实例，包含关于如何进行ViewModel进行跳转问题，还有网络请求及网络状态判断的功能点；\n\n\n## 一：关于ReactiveCocoa的知识点\n\n1：RACSigner基础知识点\n\n```obj-c\n\n信号类(RACSiganl)，只是表示当数据改变时，信号内部会发出数据，它本身不具备发送信号的能力，而是交给内部一个订阅者去发出。\n\n默认一个信号都是冷信号，也就是值改变了，也不会触发，只有订阅了这个信号，这个信号才会变为热信号，值改变了才会触发。\n\n如何订阅信号：调用信号RACSignal的subscribeNext就能订阅\n\n```\n\n常见的操作方法：\n\n```obj-c\n\nflattenMap map 用于把源信号内容映射成新的内容。\n\nconcat 组合 按一定顺序拼接信号，当多个信号发出的时候，有顺序的接收信号\n\nthen 用于连接两个信号，当第一个信号完成，才会连接then返回的信号。\n\nmerge 把多个信号合并为一个信号，任何一个信号有新值的时候就会调用\n\nzipWith 把两个信号压缩成一个信号，只有当两个信号同时发出信号内容时，并且把两个信号的内容合并成一个元组，才会触发压缩流的next事件。\n\ncombineLatest:将多个信号合并起来，并且拿到各个信号的最新的值,必须每个合并的signal至少都有过一次sendNext，才会触发合并的信号。\n\nreduce聚合:用于信号发出的内容是元组，把信号发出元组的值聚合成一个值\n\nfilter:过滤信号，使用它可以获取满足条件的信号.\n\nignore:忽略完某些值的信号.\n\ndistinctUntilChanged:当上一次的值和当前的值有明显的变化就会发出信号，否则会被忽略掉。\n\ntake:从开始一共取N次的信号\n\ntakeLast:取最后N次的信号,前提条件，订阅者必须调用完成，因为只有完成，就知道总共有多少信号.\n\ntakeUntil:(RACSignal *):获取信号直到某个信号执行完成\n\nskip:(NSUInteger):跳过几个信号,不接受。\n\nswitchToLatest:用于signalOfSignals（信号的信号），有时候信号也会发出信号，会在signalOfSignals中，获取signalOfSignals发送的最新信号。\n\ndoNext: 执行Next之前，会先执行这个Block\n\ndoCompleted: 执行sendCompleted之前，会先执行这个Block\n\ntimeout：超时，可以让一个信号在一定的时间后，自动报错。\n\ninterval 定时：每隔一段时间发出信号\n\ndelay 延迟发送next。\n\nretry重试 ：只要失败，就会重新执行创建信号中的block,直到成功.\n\nreplay重放：当一个信号被多次订阅,反复播放内容\n\nthrottle节流:当某个信号发送比较频繁时，可以使用节流，在某一段时间不发送信号内容，过了一段时间获取信号的最新内容发出。\n\n```\n\n2：RACSubject基础知识点\n\n```obj-c\nRACSubject:信号提供者，自己可以充当信号，又能发送信号  使用场景:通常用来代替代理，有了它，就不必要定义代理了\n\nRACSubject使用步骤\n1.创建信号 [RACSubject subject]，跟RACSiganl不一样，创建信号时没有block。\n2.订阅信号 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock\n3.发送信号 sendNext:(id)value\n\nRACSubject:底层实现和RACSignal不一样。\n1.调用subscribeNext订阅信号，只是把订阅者保存起来，并且订阅者的nextBlock已经赋值了。\n2.调用sendNext发送信号，遍历刚刚保存的所有订阅者，一个一个调用订阅者的nextBlock。\n\nRACSubject实例进行map操作之后, 发送完毕一定要调用-sendCompleted, 否则会出现内存泄漏; 而RACSignal实例不管是否进行map操作, 不管是否调用-sendCompleted, 都不会出现内存泄漏.\n原因 : 因为RACSubject是热信号, 为了保证未来有事件发生的时候, 订阅者可以收到信息, 所以需要对持有订阅者!\n\n```\n\n3：RACSequence基础知识点\n\n```obj-c\n\nRACSequence:RAC中的集合类，用于代替NSArray,NSDictionary,可以使用它来快速遍历数组和字典\n\n通过RACSequence对数组进行操作\n这里其实是三步\n第一步: 把数组转换成集合RACSequence numbers.rac_sequence\n第二步: 把集合RACSequence转换RACSignal信号类,numbers.rac_sequence.signal\n第三步: 订阅信号，激活信号，会自动把集合中的所有值，遍历出来。\n\n```\n\n4：RACCommand基础知识点\n\n```obj-c\n\nRACCommand:RAC中用于处理事件的类，可以把事件如何处理,事件中的数据如何传递，包装到这个类中，他可以很方便的监控事件的执行过程\n\n一、RACCommand使用步骤:\n1.创建命令 initWithSignalBlock:(RACSignal * (^)(id input))signalBlock\n2.在signalBlock中，创建RACSignal，并且作为signalBlock的返回值\n3.执行命令 - (RACSignal *)execute:(id)input\n\n二、RACCommand使用注意:\n1.signalBlock必须要返回一个信号，不能传nil.\n2.如果不想要传递信号，直接创建空的信号[RACSignal empty];\n3.RACCommand中信号如果数据传递完，必须调用[subscriber sendCompleted]，这时命令才会执行完毕，否则永远处于执行中。\n4.RACCommand需要被强引用，否则接收不到RACCommand中的信号，因此RACCommand中的信号是延迟发送的。\n\n三、RACCommand设计思想：内部signalBlock为什么要返回一个信号，这个信号有什么用。\n1.在RAC开发中，通常会把网络请求封装到RACCommand，直接执行某个RACCommand就能发送请求。\n2.当RACCommand内部请求到数据的时候，需要把请求的数据传递给外界，这时候就需要通过signalBlock返回的信号传递了。\n\n四、如何拿到RACCommand中返回信号发出的数据。\n1.RACCommand有个执行信号源executionSignals，这个是signal of signals(信号的信号),意思是信号发出的数据是信号，不是普通的类型。\n2.订阅executionSignals就能拿到RACCommand中返回的信号，然后订阅signalBlock返回的信号，就能获取发出的值。\n\n五、监听当前命令是否正在执行executing\n\n六、使用场景,监听按钮点击，网络请求\n\n```\n\n5：RACMulticastConnection基础知识点\n\n```obj-c\n\nRACMulticastConnection:用于当一个信号，被多次订阅时，为了保证创建信号时，避免多次调用创建信号中的block，造成副作用，可以使用这个类处理\n使用注意:RACMulticastConnection通过RACSignal的-publish或者-muticast:方法创建.\n\nRACMulticastConnection使用步骤:\n1.创建信号 + (RACSignal *)createSignal:(RACDisposable * (^)(id\u003cRACSubscriber\u003e subscriber))didSubscribe\n2.创建连接 RACMulticastConnection *connect = [signal publish];\n3.订阅信号,注意：订阅的不在是之前的信号，而是连接的信号。 [connect.signal subscribeNext:nextBlock]\n4.连接 [connect connect]\n\nRACMulticastConnection底层原理:\n1.创建connect，connect.sourceSignal -\u003e RACSignal(原始信号)  connect.signal -\u003e RACSubject\n2.订阅connect.signal，会调用RACSubject的subscribeNext，创建订阅者，而且把订阅者保存起来，不会执行block。\n3.[connect connect]内部会订阅RACSignal(原始信号)，并且订阅者是RACSubject\n3.1.订阅原始信号，就会调用原始信号中的didSubscribe\n3.2 didSubscribe，拿到订阅者调用sendNext，其实是调用RACSubject的sendNext\n4.RACSubject的sendNext,会遍历RACSubject所有订阅者发送信号。\n4.1 因为刚刚第二步，都是在订阅RACSubject，因此会拿到第二步所有的订阅者，调用他们的nextBlock\n\n\n需求：假设在一个信号中发送请求，每次订阅一次都会发送请求，这样就会导致多次请求。\n解决：使用RACMulticastConnection就能解决.\n\n```\n\n6：RAC结合UI一般事件\n\n```obj-c\n\nrac_signalForSelector : 代替代理\n\nrac_valuesAndChangesForKeyPath: KVO\n\nrac_signalForControlEvents:监听事件\n\nrac_addObserverForName 代替通知\n\nrac_textSignal：监听文本框文字改变\n\nrac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组)，每一个signal都至少sendNext过一次，就会去触发第一个selector参数的方法。\n\n```\n\n7：高阶操作知识内容\n\n\n8：RAC并发编程知识点\n\n\n```obj-c\n\n1: subscribeOn运用\n\n    RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id\u003cRACSubscriber\u003e subscriber) {\n        NSLog(@\"%@ 111\",[NSThread currentThread]);\n        \n        //可以放更新UI操作\n        \n        [subscriber sendNext:@0.1];\n        RACDisposable *disposable = [[RACScheduler scheduler] schedule:^{\n            NSLog(@\"%@ 5555\",[NSThread currentThread]);\n            [subscriber sendNext:@1.1];\n            [subscriber sendCompleted];\n        }];\n        return disposable;\n    }];\n    [[RACScheduler scheduler] schedule:^{\n        NSLog(@\"%@ 222\",[NSThread currentThread]);\n        [[signal subscribeOn:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {\n            NSLog(@\"%@ %@\",[NSThread currentThread], x);\n        }]; }];\n    NSLog(@\"%@ 4444\",[NSThread currentThread]);\n\n//使用subscribeOn 可以让signal内的代码在主线程中运行，sendNext在哪个线程 则对应的订阅输出就在对应线程上，所以0.1输出是在主线程中； 所以当在signal里面可能要放一些更新UI的操作，而这些是要在主线程才能处理，而订阅者却无法确认，所以要使用subscribeOn让它在主线程中；\n//能够保证didSubscribe block在指定的scheduler\n//不能保证sendNext、 error、 complete在哪个scheduler\n\n\n2：deliverOn运用\n\n    RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id\u003cRACSubscriber\u003e subscriber) {\n        NSLog(@\"%@ 111\",[NSThread currentThread]);\n        [subscriber sendNext:@0.1];\n        RACDisposable *disposable = [[RACScheduler scheduler] schedule:^{\n            NSLog(@\"%@ 555\",[NSThread currentThread]);\n            [subscriber sendNext:@1.1];\n            [subscriber sendCompleted];\n        }];\n        return disposable;\n    }];\n    [[RACScheduler scheduler] schedule:^{\n        NSLog(@\"%@ 222\",[NSThread currentThread]);\n        [[signal deliverOn:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {\n            NSLog(@\"%@ %@\",[NSThread currentThread], x);\n            \n            //可以放UI更新操作\n            \n        }]; }];\n\n//当我们让订阅的处理代码在指定的线程中执行，而不必去关心发送信号的当前线程，就可以deliverOn\n```\n\n\n9：冷信号跟热信号知识点\n\n```obj-c\n\nHot Observable是主动的，尽管你并没有订阅事件，但是它会时刻推送，就像鼠标移动；而Cold Observable是被动的，只有当你订阅的时候，它才会发布消息。\n\nHot Observable可以有多个订阅者，是一对多，集合可以与订阅者共享信息；而Cold Observable只能一对一，当有不同的订阅者，消息是重新完整发送。\n\n热信号是主动的，即使你没有订阅事件，它仍然会时刻推送 而冷信号是被动的，只有当你订阅的时候，它才会发送消息\n热信号可以有多个订阅者，是一对多，信号可以与订阅者共享信息 而冷信号只能一对一，当有不同的订阅者，消息会从新完整发送\n\n冷信号与热信号的本质区别在于是否保持状态，冷信号的多次订阅是不保持状态的，而热信号的多次订阅可以保持状态\n\n\n```\n\n10：RACDisposable知识点\n\n```obj-c\n\nRACDisposable用于取消订阅信号，默认信号发送完之后就会主动的取消订阅。订阅信号使用的subscribeNext:方法返回的就是RACDisposable类型的对象\n\n当订阅者发送信号- (void)sendNext:(id)value之后，会执行：- (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock中的nextBlock。当nextBlock执行完毕也就意味着subscribeNext方法返回了RACDisposable对象。\n\n1.如果不强引用订阅者对象，默认情况下会自动取消订阅，我们可以拿到RACDisposable 用+ (instancetype)disposableWithBlock:(void (^)(void))block做清空资源的一些操作了。\n\n2.如果不希望自动取消订阅，我们应该强引用RACSubscriber * subscriber。在想要取消订阅的时候用- (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock返回的RACDisposable对象去调用- (void)dispose方法\n\n```\n\n11：RACChannel知识点\n\n```obj-c\n    RACChannelTerminal *channelA = RACChannelTo(self, valueA);\n    RACChannelTerminal *channelB = RACChannelTo(self, valueB);\n    [[channelA map:^id(NSString *value) {\n        if ([value isEqualToString:@\"西\"]) {\n            return @\"东\";\n        }\n        return value;\n    }] subscribe:channelB];\n    [[channelB map:^id(NSString *value) {\n        if ([value isEqualToString:@\"左\"]) {\n            return @\"右\";\n        }\n        return value;\n    }] subscribe:channelA];\n    [[RACObserve(self, valueA) filter:^BOOL(id value) {\n        return value ? YES : NO;\n    }] subscribeNext:^(NSString* x) {\n        NSLog(@\"你向%@\", x);\n    }];\n    [[RACObserve(self, valueB) filter:^BOOL(id value) {\n        return value ? YES : NO;\n    }] subscribeNext:^(NSString* x) {\n        NSLog(@\"他向%@\", x);\n    }];\n    self.valueA = @\"西\";\n    self.valueB = @\"左\";\n    \n    \n    RACChannelTerminal *characterRemainingTerminal = RACChannelTo(_loginButton, titleLabel.text);\n    \n    [[self.userNameText.rac_textSignal map:^id(NSString *text) {\n        return [@(100 - (NSInteger)text.length) stringValue];\n    }] subscribe:characterRemainingTerminal];\n\n```\n\n\n12：RAC倒计时小实例\n\n```obj-c\n    //倒计时的效果\n    RACSignal *(^counterSigner)(NSNumber *count)=^RACSignal *(NSNumber *count)\n    {\n        RACSignal *timerSignal=[RACSignal interval:1 onScheduler:RACScheduler.mainThreadScheduler];\n        RACSignal *counterSignal=[[timerSignal scanWithStart:count reduce:^id(NSNumber *running, id next) {\n            return @(running.integerValue -1);\n        }] takeUntilBlock:^BOOL(NSNumber *x) {\n            return x.integerValue\u003c0;\n        }];\n        \n        return [counterSignal startWith:count];\n    };\n    \n    \n    RACSignal *enableSignal=[self.myTextField.rac_textSignal map:^id(NSString *value) {\n        return @(value.length==11);\n    }];\n    \n    RACCommand *command=[[RACCommand alloc]initWithEnabled:enableSignal signalBlock:^RACSignal *(id input) {\n        return counterSigner(@10);\n    }];\n    \n    RACSignal *counterStringSignal=[[command.executionSignals switchToLatest] map:^id(NSNumber *value) {\n        return [value stringValue];\n    }];\n    \n    RACSignal *resetStringSignal=[[command.executing filter:^BOOL(NSNumber *value) {\n        return !value.boolValue;\n    }] mapReplace:@\"点击获得验证码\"];\n    \n    //[self.myButton rac_liftSelector:@selector(setTitle:forState:) withSignals:[RACSignal merge:@[counterStringSignal,resetStringSignal]],[RACSignal return:@(UIControlStateNormal)],nil];\n    \n    //上面也可以写成下面这样\n    @weakify(self);\n    [[RACSignal merge:@[counterStringSignal,resetStringSignal]] subscribeNext:^(id x) {\n        @strongify(self);\n        [self.myButton setTitle:x forState:UIControlStateNormal];\n    }];\n    \n    self.myButton.rac_command=command;\n    \n    \n    //编写关于委托的编写方式 是在self上面进行rac_signalForSelector\n    [[self\n      rac_signalForSelector:@selector(textFieldShouldReturn:)\n      fromProtocol:@protocol(UITextFieldDelegate)]\n    \tsubscribeNext:^(RACTuple *tuple) {\n            @strongify(self)\n            if (tuple.first == self.myTextField)\n            {\n                NSLog(@\"触发\");\n            };\n        }];\n    \n    self.myTextField.delegate = self;\n\n```\n\n13：常见的宏定义运用\n\n```obj-c\n\n1：\nRAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于给某个对象的某个属性绑定\n只要文本框文字改变，就会修改label的文字\nRAC(self.labelView,text) = _textField.rac_textSignal;\n\n2:\nRACObserve(self, name):监听某个对象的某个属性,返回的是信号。\n[RACObserve(self.view, center) subscribeNext:^(id x) {\n    NSLog(@\"%@\",x);\n}];\n\n\n当RACObserve放在block里面使用时一定要加上weakify，不管里面有没有使用到self；否则会内存泄漏，因为RACObserve宏里面就有一个self\n@weakify(self);\nRACSignal *signal3 = [anotherSignal flattenMap:^(NSArrayController *arrayController) {\n     //Avoids a retain cycle because of RACObserve implicitly referencing self\n    @strongify(self);\n    return RACObserve(arrayController, items);\n}];\n\n3:\n@weakify(Obj)和@strongify(Obj),一般两个都是配套使用,在主头文件(ReactiveCocoa.h)中并没有导入，需要自己手动导入，RACEXTScope.h才可以使用。但是每次导入都非常麻烦，只需要在主头文件自己导入就好了\n\n4:\nRACTuplePack：把数据包装成RACTuple（元组类）\n把参数中的数据包装成元组\nRACTuple *tuple = RACTuplePack(@10,@20);\n\n5:\nRACTupleUnpack：把RACTuple（元组类）解包成对应的数据\n把参数中的数据包装成元组\nRACTuple *tuple = RACTuplePack(@\"xmg\",@20);\n\n解包元组，会把元组的值，按顺序给参数里面的变量赋值\nname = @\"xmg\" age = @20\nRACTupleUnpack(NSString *name,NSNumber *age) = tuple;\n\n```\n\n\n\n## 二：关于使用ReactiveCocoa结合MVVM模式的实例；\n\nMVVM模式和MVC模式一样，主要目的是分离视图（View）和模型（Model），有几大优点\n\n1. 低耦合。视图（View）可以独立于Model变化和修改，一个ViewModel可以绑定到不同的\"View\"上，当View变化的时候Model可以不变，当Model变化的时候View也可以不变。\n\n2. 可重用性。你可以把一些视图逻辑放在一个ViewModel里面，让很多view重用这段视图逻辑。\n\n3. 独立开发。开发人员可以专注于业务逻辑和数据的开发（ViewModel），设计人员可以专注于页面设计。\n\n4. 可测试。界面素来是比较难于测试的，而现在测试可以针对ViewModel来写。\n\n\n\n\n## 三：单元测试知识\n\n单元测试这边主要采用两种方式，一种是XCode自动的XCTestCase进行，如下面这些就是它所对应的断言等，另外一种是采有KIWI的插件进行测试；项目中有针对viewController、viewModel、帮助类等的测试实例；运用快捷键（command+U）可以运行单元测试实例；\n\n\n```obj-c\n\n//知识点一：\n//方法在XCTestCase的测试方法调用之前调用，可以在测试之前创建在test case方法中需要用到的一些对象等\n//- (void)setUp ;\n//当测试全部结束之后调用tearDown方法，法则在全部的test case执行结束之后清理测试现场，释放资源删除不用的对象等\n//- (void)tearDown ;\n//测试代码执行性能\n//- (void)testPerformanceExample\n\n\n//知识点二：\n//通用断言\nXCTFail(format…)\n//为空判断，a1为空时通过，反之不通过；\nXCTAssertNil(a1, format...)\n//不为空判断，a1不为空时通过，反之不通过；\nXCTAssertNotNil(a1, format…)\n//当expression求值为TRUE时通过；\nXCTAssert(expression, format...)\n//当expression求值为TRUE时通过；\nXCTAssertTrue(expression, format...)\n//当expression求值为False时通过；\nXCTAssertFalse(expression, format...)\n//判断相等，[a1 isEqual:a2]值为TRUE时通过，其中一个不为空时，不通过；\nXCTAssertEqualObjects(a1, a2, format...)\n//判断不等，[a1 isEqual:a2]值为False时通过；\nXCTAssertNotEqualObjects(a1, a2, format...)\n//判断相等（当a1和a2是 C语言标量、结构体或联合体时使用,实际测试发现NSString也可以）；\nXCTAssertEqual(a1, a2, format...)\n//判断不等（当a1和a2是 C语言标量、结构体或联合体时使用）；\nXCTAssertNotEqual(a1, a2, format...)\n//判断相等，（double或float类型）提供一个误差范围，当在误差范围（+/-accuracy）以内相等时通过测试；\nXCTAssertEqualWithAccuracy(a1, a2, accuracy, format...)\n//判断不等，（double或float类型）提供一个误差范围，当在误差范围以内不等时通过测试；\nXCTAssertNotEqualWithAccuracy(a1, a2, accuracy, format...)\n//异常测试，当expression发生异常时通过，反之不通过；\nXCTAssertThrows(expression, format...)\n//异常测试，当expression发生specificException异常时通过；反之发生其他异常或不发生异常均不通过\nXCTAssertThrowsSpecific(expression, specificException, format...)\n//异常测试，当expression发生具体异常、具体异常名称的异常时通过测试，反之不通过；\nXCTAssertThrowsSpecificNamed(expression, specificException, exception_name, format...)\n//异常测试，当expression没有发生异常时通过测试；\nXCTAssertNoThrow(expression, format…)\n//异常测试，当expression没有发生具体异常、具体异常名称的异常时通过测试，反之不通过；\nXCTAssertNoThrowSpecific(expression, specificException, format...)\n//异常测试，当expression没有发生具体异常、具体异常名称的异常时通过测试，反之不通过\nXCTAssertNoThrowSpecificNamed(expression, specificException, exception_name, format...)\n\n```\n\n\n\n采用KiWi的单元测试效果：\n\n```obj-c\n\n#import \u003cKiwi/Kiwi.h\u003e\n//把原本在项目pch中那些第三方插件的头文件也要引入\n#import \u003cReactiveCocoa/ReactiveCocoa.h\u003e\n\n//测试LogInViewController\n#import \"RACTestLoginViewController.h\"\n\n\nSPEC_BEGIN(LoginViewControllerSpec)\n\ndescribe(@\"RACTestLoginViewController\", ^{\n    __block RACTestLoginViewController *controller = nil;\n    \n    beforeEach(^{\n        controller = [RACTestLoginViewController new];\n        [controller view];\n    });\n    \n    afterEach(^{\n        controller = nil;\n    });\n    \n    describe(@\"Root View\", ^{\n        \n        context(@\"when view did load\", ^{\n            it(@\"should bind data\", ^{\n                controller.userNameText.text=@\"wujunyang\";\n                controller.passWordTest.text=@\"123456\";\n                //\n                //一定要调用sendActionsForControlEvents方法来通知UI已经更新 因为RAC是监听这个输入框的变化\n                [controller.userNameText sendActionsForControlEvents:UIControlEventEditingChanged];\n                [controller.passWordTest sendActionsForControlEvents:UIControlEventEditingChanged];\n                \n                [[controller.myLoginViewModel.username should] equal:controller.userNameText.text];\n                [[controller.myLoginViewModel.password should] equal:controller.passWordTest.text];\n            });\n        });\n        \n    });\n});\n\nSPEC_END\n\n```\n\n关于kiwi中的操作类型可以直接查看：https://github.com/allending/Kiwi/wiki/Expectations\n\n\n注意：发现在进行单元测试时，针对RAC就会报[RACStream(Operations) reduceEach:]_block_invoke，后来发现是Pod引入写法有问题，导致的【it usually means RAC is being linked twice. Make sure it's only in your app target.】 所以测试的MobileProjectTests特别要注意；\n\n```obj-c\n\nplatform :ios, '7.0'\n\nabstract_target 'MobileProjectDefault' do\n    pod 'AFNetworking', '~\u003e2.6.0'\n    pod 'SDWebImage', '~\u003e3.7'\n    pod 'JSONModel', '~\u003e 1.0.1'\n    pod 'Masonry','~\u003e0.6.1'\n    pod 'FMDB/common' , '~\u003e2.5'\n    pod 'FMDB/SQLCipher', '~\u003e2.5'\n    pod 'CocoaLumberjack', '~\u003e 2.0.0-rc'\n    pod 'ReactiveCocoa', '2.5'\n    pod 'CYLTabBarController'\n    pod 'MLeaksFinder'  #可以把它放在MobileProject_Local的target中 这样就不会影响到产品环境\n    pod 'RealReachability'\n    \n    target 'MobileProject_Local' do\n        \n    end\n    \n    target 'MobileProject' do\n        \n        target 'MobileProjectTests' do\n            inherit! :search_paths\n            pod 'Kiwi', '~\u003e 2.3.1'\n        end\n    end\nend\n\n```\n\n\n## 四：项目效果：\n\n\n\u003cimg src=\"https://github.com/wujunyang/MVVMReactiveCocoaDemo/blob/master/MobileProject/3.gif\" width=200px height=300px\u003e\u003c/img\u003e\n\n\n\n## 五：ReactiveCocoa知识分享地址\n\n\n```obj-c\n\nReactiveCocoa 和 MVVM 入门 http://yulingtianxia.com/blog/2015/05/21/ReactiveCocoa-and-MVVM-an-Introduction/\n\nMVVM Tutorial with ReactiveCocoa  http://southpeak.github.io/blog/2014/08/08/mvvmzhi-nan-yi-:flickrsou-suo-shi-li/\n\nReactiveCocoa 1-官方readme文档翻译  http://cindyfn.com/reactivecocoa/2014/12/01/ios-frame-use-ReactiveCocoa.html\n\n这样好用的ReactiveCocoa，根本停不下来  http://www.cocoachina.com/ios/20150817/13071.html\n\nReactiveCocoa基本组件：深入浅出RACCommand  http://www.tuicool.com/articles/nYJRvu\n\nReactiveCocoa自述：工作原理和应用  http://www.cocoachina.com/ios/20150702/12302.html\n\nRACSignal的巧克力工厂 http://www.cnblogs.com/sunnyxx/p/3547763.html\n\nReactiveCocoa一些概念讲解  http://www.thinksaas.cn/group/topic/347067/\n\n细说ReactiveCocoa的冷信号与热信号（二）：为什么要区分冷热信号  http://www.tuicool.com/articles/e2uMzyq\n\n细说ReactiveCocoa的冷信号与热信号（三）：怎么处理冷信号与热信号  http://www.tuicool.com/articles/emIVZjY\n\n最快让你上手ReactiveCocoa之基础篇   http://www.jianshu.com/p/87ef6720a096\n\n最快让你上手ReactiveCocoa之进阶篇  http://www.jianshu.com/p/e10e5ca413b7\n\nReactiveCocoa基础：理解并使用RACCommand http://www.yiqivr.com/2015/10/19/%E8%AF%91-ReactiveCocoa%E5%9F%BA%E7%A1%80%EF%BC%9A%E7%90%86%E8%A7%A3%E5%B9%B6%E4%BD%BF%E7%94%A8RACCommand/  \n\nRAC一些代码总结：https://github.com/shuaiwang007/RAC \n\nReactiveCocoa小总结   http://www.jianshu.com/p/8fd6c8349774\n\n如何在ReactiveCocoa中写单元测试   http://www.jianshu.com/p/412875512bd1\n\nTDD的iOS开发初步以及Kiwi使用入门 https://onevcat.com/2014/02/ios-test-with-kiwi/\n\n```\n\n# 订阅号\n\n最近有个妹子弄的一个关于扩大眼界跟内含的订阅号，每天都会更新一些深度内容，在这里如果你感兴趣也可以关注一下，当然可以关注后输入数字：5  会有我的微信号，如果有问题你也可以在那找到我；当然不感兴趣无视此信息；\n\n\n\u003cimg src=\"https://github.com/wujunyang/jiaModuleDemo/blob/master/jiaModuleDemo/ProjectImage/dy.jpg\" width=200px height=200px\u003e\u003c/img\u003e\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fwujunyang%2Fmvvmreactivecocoademo","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Fwujunyang%2Fmvvmreactivecocoademo","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fwujunyang%2Fmvvmreactivecocoademo/lists"}