什么是RAC:
git 重量型開源項目,主要是針對各種事件的處理 拌倍。
什么是響應(yīng)式編程:
一個簡單的理解:如果a + b =c 修改 a 在 修改 b c會變嗎暇番?結(jié)果是不會變的 于个,如果c 變是需要再次調(diào)用 a + b = c 這個計算镐作,響應(yīng)式編程:就是修改a 或者 b 的時候 c就會 立即變換锈津。所以說:響應(yīng)式編程就是 在事件發(fā)生變換的時候立即做出相應(yīng).
IOS 開發(fā)中有哪些事件發(fā)生:
tagrget
delegate
kvo
通知
時鐘(NSTime)
網(wǎng)絡(luò)異步回調(diào)
RAC家族:4大家族
pod search ReactiveObjC
RACSignal:
具體使用: 簡稱《信號3部曲》
/*
信號:
1:創(chuàng)建信號 :(冷信號)
2:訂閱信號:(熱信號)
3:發(fā)送信號:
*/
//? ? 創(chuàng)建信號(冷信號)
RACSignal *signal =? [RACSignal createSignal:^RACDisposable *(id subscriber) {
//? ? ? ? 發(fā)送信號
[subscriber sendNext:@"this is signal"];
return nil;
}];
//? ? 訂閱信號? (subscribe) 訂閱(熱信號)
[signal subscribeNext:^(id x) {
//? ? ? x :信號內(nèi)容
NSLog(@"x is value :%@ ",x);
}];
構(gòu)造方法分析
+ (RACSignal *)createSignal:(RACDisposable * (^)(id subscriber))didSubscribe {
return [RACDynamicSignal createSignal:didSubscribe];
}
RACDynamicSignal:動態(tài)信號
看看這個方法:
[RACDynamicSignal createSignal:didSubscribe]
+ (RACSignal *)createSignal:(RACDisposable * (^)(id subscriber))didSubscribe {
RACDynamicSignal *signal = [[self alloc] init];
signal->_didSubscribe = [didSubscribe copy];
return [signal setNameWithFormat:@"+createSignal:"];
}
結(jié)論:創(chuàng)建信號的時候干了2件事情
1:創(chuàng)建了RACDynamicSignal
2:保存一個block didSubscribe
創(chuàng)建信號的內(nèi)部處理:
//? ? 創(chuàng)建冷信號
RACSignal *signal = [RACSignal createSignal:^RACDisposable * _Nullable(id? _Nonnull subscriber) {
NSLog(@"創(chuàng)建信號");
[subscriber sendNext:@"發(fā)送信號已經(jīng)收到"];
NSLog(@"發(fā)送信號");
return nil;
}];
//? ? 訂閱信號 (熱信號)
[signal subscribeNext:^(id? _Nullable x) {
//? ? ? ? x:信號內(nèi)容
NSLog(@"this is %@",x);
NSLog(@"我訂閱了信號");
}];
1:測試: 將訂閱信號注釋掉 查看日志 結(jié)果是什么都沒有 so 假設(shè)結(jié)論:創(chuàng)建信號必須先訂閱
2:測試:將發(fā)送信號代碼去掉:結(jié)果 :打印出創(chuàng)建信號log so 假設(shè)結(jié)論:我要訂閱信號必須先發(fā)送
訂閱信號的內(nèi)部處理
//? ? 訂閱信號 (熱信號)
[signal subscribeNext:^(id? _Nullable x) {
//? ? ? ? x:信號內(nèi)容
NSLog(@"this is %@",x);
NSLog(@"我訂閱了信號");
}];
進(jìn)入 subscribeNext :
- (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock {
NSCParameterAssert(nextBlock != NULL);
RACSubscriber *o = [RACSubscriber subscriberWithNext:nextBlock error:NULL completed:NULL];
return [self subscribe:o];
}
*記錄下nextBlock
在進(jìn)入:subscriberWithNext: 發(fā)現(xiàn)創(chuàng)建了RACSubscriber 點擊進(jìn)入 subscribe:選擇 RACDynamicSignal 的方法呀酸,發(fā)現(xiàn) 在subscribe方法中執(zhí)行了,self.didSubscribe(subscriber) 這里解釋了:創(chuàng)建信號必須先訂閱(不訂閱 block 就不執(zhí)行)
如果 我要訂閱信號必須先發(fā)送
- (void)sendNext:(id)value {
@synchronized (self) {
void (^nextBlock)(id) = [self.next copy];
if (nextBlock == nil) return;
nextBlock(value);
}
}
執(zhí)行nextBlock
具體的流程如下:
RAC 的簡單使用
KVO的簡單使用
kvo 的使用實現(xiàn)對_p對象的 name 屬性監(jiān)聽
//? ? kvo
_p = [[Person alloc]init];
//? ? 對P監(jiān)聽
[RACObserve(self.p, name)subscribeNext:^(id? _Nullable x) {
NSLog(@"x is value? %@",x);
}];
點擊屏幕修改p的name值
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
self.p.name = [NSString stringWithFormat:@"mrlee %05d",arc4random_uniform(200)];
}
target使用
一句話搞定 創(chuàng)建 訂閱 發(fā)送 三部曲
[[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"x is value %@",x);
}];
我們點擊按鈕顯示x的值為:
x is value button
拿個這個是不是想干啥 就干啥~
輸入框點擊監(jiān)聽
[[self.textField rac_textSignal]subscribeNext:^(NSString * _Nullable x) {
NSLog(@"x is value %@",x);
}
通知
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidEnterBackgroundNotification object:nil]subscribeNext:^(NSNotification * _Nullable x) {
NSLog(@"ios is value %@",x);
}];
RAC中的坑
循環(huán)引用
1.制作循環(huán)引用案例
首先 做a push b 頁面 在b頁面添加如下代碼
-(void)dealloc{
NSLog(@"bay bay !");
}
在從b pop 回 a 打印出bay bay ! 說明VC 釋放
修改下
- (void)viewDidLoad {
[super viewDidLoad];
[self demo2];
}
-(void)demo2{
//? ? 信號的生成
[[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"x is value %@",x);
self.textField.text =@"你好~";
}];
}
點擊按鈕 在pop 回 a 沒有走bay bay ! 說明循環(huán)引用
修改下
//? ? 信號的生成
__weak typeof (self) weakSelf = self;
[[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"x is value %@",x);
weakSelf.textField.text =@"你好~";
}];
打印bay bay
RAC 提供了相關(guān)解決方案 @weakify(self);? @strongify(self);
//? ? 信號的生成
@weakify(self);
[[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(__kindof UIControl * _Nullable x) {
@strongify(self);
NSLog(@"x is value %@",x);
self.textField.text =@"你好~";
}];
是如何造成循環(huán)引用的
self.btn 對 self.view.subview 進(jìn)行強(qiáng)引用 self.textField 是self.btn 對其強(qiáng)引用 因為 他是在self.btn的Block中 self.textField 又被self.view.subview 進(jìn)行強(qiáng)引用 so 循環(huán)引用
這個說明:只要在rac中用了self 就會造成循環(huán)引用琼梆,解決辦法 @weakify 和 @strongify
Command命令
//? ? 創(chuàng)建命令
RACCommand *command = [[RACCommand alloc]initWithSignalBlock:^RACSignal * _Nonnull(id? _Nullable input) {
return nil;
}];
//? ? 執(zhí)行命令
[command execute:@"xx"];
運行這段代碼 會發(fā)現(xiàn) app carsh 掉了 原因是因為 return nil; 應(yīng)該返回一個信號
修改后
//? ? 創(chuàng)建命令
RACCommand *command = [[RACCommand alloc]initWithSignalBlock:^RACSignal * _Nonnull(id? _Nullable input) {
return [RACSignal createSignal:^RACDisposable * _Nullable(id? _Nonnull subscriber) {
return nil;
}];
}];
//? ? 執(zhí)行命令
[command execute:@"xx"];
input:執(zhí)行命令的內(nèi)容 (輸入的指令)這樣就不會崩潰了但是一個流程還沒有走完 還差個接收 一個完整的流程如下
//? ? 創(chuàng)建命令
RACCommand *command = [[RACCommand alloc]initWithSignalBlock:^RACSignal * _Nonnull(id? _Nullable input) {
//? ? ? ? input 執(zhí)行命令里的內(nèi)容
return [RACSignal createSignal:^RACDisposable * _Nullable(id? _Nonnull subscriber) {
[subscriber sendNext:@"我是從命令里面發(fā)送的消息"];
return nil;
}];
}];
//? ? 執(zhí)行命令
[[command execute:@"xx"] subscribeNext:^(id? _Nullable x) {
NSLog(@"x is value %@",x);
}];
MVVM
具體是什么是mvvm 我想不用說也有很多的資料講解 性誉。這里直接就不多說了,只說如下幾點
1:MVVM :就是MVC的瘦身劑
2:MVVM 方便功能測試
下面做個簡單的登錄頁面 :需求 如果沒有輸入 userName 或者pwd 登錄按鈕是無法點擊
具體實現(xiàn)如下:
//? ? 綁定
//? ? 多個信號綁定成一個信號
[[RACSignal combineLatest:@[self.userName.rac_textSignal,self.passWord.rac_textSignal] reduce:^id _Nullable(NSString *userName,NSString *pwd){
return @(userName.length && pwd.length);
}]subscribeNext:^(id? _Nullable x) {
NSLog(@"x is value %@",x);
}];
1使用:combineLatest reduce 進(jìn)行多個信號的綁定
// RAC(self.loginBtn,enabled) 監(jiān)聽 UI 的狀態(tài)? 整合后
RAC(self.loginBtn,enabled) =? [RACSignal combineLatest:@[self.userName.rac_textSignal,self.passWord.rac_textSignal] reduce:^id _Nullable(NSString *userName,NSString *pwd){
return @(userName.length && pwd.length);
}];
OK 搞定 么又看出就這一句代碼搞定 判斷 username 和 pwd都必須都有值 才可以點點擊按鈕
RAC(target,...) 用于監(jiān)聽 一個對象的相關(guān)屬性 返回的為 RACSignal
combineLatest 綁定多個有共性的信號
demo地址如果對您有用請點個star