我们这里首先用一个简单的登录页面来说明一下MVVM构架,之后再用一个网络请求深入一下MVVM
- 1, 如果不使用MVVM构架,把所有的东西都写在控制器中,比较臃肿,如下:
- (void)viewDidLoad {
[super viewDidLoad];
//1,登录按钮可否点击的业务逻辑
RACSignal *combineSignal = [RACSignal combineLatest:@[_account.rac_textSignal, _pwd.rac_textSignal] reduce:^id(NSString *account, NSString *pwd){
return @(account.length && pwd.length);
}];
/*
[combineSignal subscribeNext:^(id x) {
_login.enabled = [x boolValue];
}];
*/
RAC(_login,enabled) = combineSignal;
//2,把点击登录按钮逻辑抽离出来
RACCommand *loginCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
NSLog(@"在这里请求数据---%@----%@",_account.text ,_pwd.text);
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
[subscriber sendNext:@"把请求到的数据发送出去--在外面进行调用"];
[subscriber sendCompleted];//这一步表示信息发送完成,之后调用会重新开始并发送信息,如果没有这一句,下一次就不会发送消息出去
return nil;
}];
}];
[loginCommand.executionSignals.switchToLatest subscribeNext:^(id x) {
NSLog(@"在外部拿到的数据-------%@", x);
}];
//3, 监听登录按钮点击,执行命令的业务逻辑即可
[[_login rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"点击了登录按钮");
//点击的时候进行command的调用
[loginCommand execute:nil];
}];
}
- 2,使用VMMV进行构架
- 控制器中代码只需要如下:
-(LoginViewModel *)loginVM{
if(_loginVM == nil){
_loginVM = [[LoginViewModel alloc] init];
}
return _loginVM;
}
- (void)viewDidLoad {
[super viewDidLoad];
// mvvm
RAC(self.loginVM, account) = _account.rac_textSignal;
RAC(self.loginVM, pwd) = _pwd.rac_textSignal;
RAC(self.login, enabled) = _loginVM.loginEnableSignal;
[[_login rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"点击了登录按钮");
[_loginVM.loginCommand execute:nil];
}];
[_loginVM.loginCommand.executionSignals.switchToLatest subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
}
- 其余代码都抽取到VM中了:
- (instancetype)init{
if (self = [super init]) {
//1,登录按钮属性
//RACObserve是观察者宏, 一旦被观察内容发生变化就发出信号
_loginEnableSignal = [RACSignal combineLatest:@[RACObserve(self, account), RACObserve(self, pwd)] reduce:^id(NSString *account, NSString *pwd){
return @(account.length && pwd.length);
}];
//2,登录按钮事件
_loginCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
NSLog(@"在这里请求数据---%@----%@",self.account, self.pwd);
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
[subscriber sendNext:@"把请求到的数据发送出去--在外面进行调用"];
[subscriber sendCompleted];//这一步表示信息发送完成,之后调用会重新开始并发送信息,如果没有这一句,下一次就不会发送消息出去
return nil;
}];
}];
}
return self;
}
3, 使用MVVM处理网络登录逻辑
首先控制器中只需要如下代码:
//如果想要在外面拿到数据,需要先订阅
[self.requestVM.requestCommand.executionSignals.switchToLatest subscribeNext:^(id x) {
NSLog(@"----%@",x);
}];
[self.requestVM.requestCommand execute:nil];
- 其他逻辑代码都放在VM中,需要注意的是,在存放逻辑代码的时候,为了把请求到的数据传递出去,需要把请求代码放在信号中,通过订阅者发送出去,然后在命令中把信号返回就可以了。
- (RACCommand *)requestCommand{
if (_requestCommand == nil) {
_requestCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
NSLog(@"请求网络数据");
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
[manager GET:@"https://api.douban.com/v2/book/search" parameters:@{@"q":@"美女"} progress:^(NSProgress * _Nonnull downloadProgress) {
} success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
NSLog(@"zhangdanfeng");
[subscriber sendNext:responseObject];
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
}];
return nil;
}];
return signal;
}];
}
return _requestCommand;
}