KVO方法中的参数
全称: Key Value Observing
在ios里,可以添加观察者模式,来实现某个property更改后,通知指定的类。
然后到observeValueForKeyPath:ofObject:change:context:提供处理
[object addObserver: observer forKeyPath: @"frame" options: 0 context: nil];
调用方法是里:
object : 被观察对象
observer: 观察对象
forKeyPath里面带上property的name,如UIView的frame、center等等
options: 有4个值,分别是:
NSKeyValueObservingOptionNew 把更改之后的值提供给处理方法
NSKeyValueObservingOptionOld 把更改之前的值提供给处理方法
NSKeyValueObservingOptionInitial 把初始化的值提供给处理方法,一旦注册,立马就会调用一次。通常它会带有新值,而不会带有旧值。
NSKeyValueObservingOptionPrior 分2次调用。在值改变之前和值改变之后。
注:例子里的0就代表不带任何参数进去
context: 可以带入一些参数,其实这个挺好用的,任何类型都可以,自己强转就好了。
处理方法里:
keyPath: 对应forKeyPath
object: 被观察的对象
change: 对应options里的NSKeyValueObservingOptionNew、NSKeyValueObservingOptionOld等
context: 对应context
我的例子
假设对象A中有属性B(对象),对象B中有属性C。
在A中观察属性B中的属性C的变化。
登录前需要检查app的版本是否可用,新建一个检查app版本的类AppVersionManager,其中有一个方法reqVersion用来发请求,判断当前版本是否可用。
.h
#import <Foundation/Foundation.h>
typedef NS_ENUM(NSInteger, AppVersionState) {
AppVersionStateAvailableWithoutNewVersion,
AppVersionStateAvailableWithNewVersion,
AppVersionStateUnAvailable,
AppVersionStateReqFailed,
AppVersionStateNull,
};
@interface AppVersionManager : NSObject
@property (nonatomic) AppVersionState versionState;
- (void)reqVersion;
@end
.m
#import "AppVersionManager.h"
@implementation AppVersionManager
- (instancetype)init {
if (self = [super init]) {
self.versionState = AppVersionStateNull;
}
return self;
}
- (void)reqVersion
{
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
//省略...
[manager POST:urlString parameters:params progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
NSInteger versionState = [(NSDictionary *)responseObject[@"versionState"] intValue];
switch (versionState) {
case 0:
self.versionState = AppVersionStateAvailableWithoutNewVersion;
break;
case 1:
self.versionState = AppVersionStateAvailableWithNewVersion;
break;
case 2:
self.versionState = AppVersionStateUnAvailable;
break;
default:
self.versionState = AppVersionStateReqFailed;
break;
}
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
self.versionState = AppVersionStateReqFailed;
}];
}
@end
然后在登录LoginVC中声明一个属性versionManager,使用KVO观察versionManager中属性versionState的变化,
@property (nonatomic, strong) AppVersionManager *versionManager;
- (void)viewDidLoad {
[super viewDidLoad];
self.versionManager = [[AppVersionManager alloc] init];
//添加观察,观察新值的变化
[self.versionManager addObserver:self forKeyPath:@"versionState" options:NSKeyValueObservingOptionNew context:nil];
//请求版本,当请求回调给versionState赋新值时,会调用observeValueForKeyPath方法
[self.versionManager reqVersion];
}
//重写了observeValueForKeyPath:ofObject:change:context:方法,属性改变时会执行此方法。
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context
{
if ([keyPath isEqualToString:@"versionState"]) {
//如果属性versionState发生改变,则执行方法checkVersion在界面上做出相应提示或操作
[self checkVersion];
}
}
//检查版本的结果
- (void)checkVersion
{
switch (self.versionManager.versionState) {
case AppVersionStateAvailableWithoutNewVersion:
[self login];
break;
case AppVersionStateAvailableWithNewVersion:
[self login];
break;
case AppVersionStateUnAvailable:
self.warningLabel.text = @"sorry,当前版本不可用\n请前往AppStore更新到最新版本";
break;
case AppVersionStateReqFailed:
self.warningLabel.text = @"sorry,检查版本失败";
break;
case AppVersionStateNull:
self.warningLabel.text = @"sorry,检查版本失败";
break;
default:
break;
}
}
最后,得在dealloc方法中移除观察对象!
- (void)dealloc{
[self.versionManager removeObserver:self forKeyPath:@"versionState"];
}
方法[observeValueForKeyPath:ofObject:change:context:]里的change这个NSDictionary对象包含了相应的值,需要的话可以取到该对象的新值和旧值。如:[change valueForKey:NSKeyValueChangeNewKey]
需要强调的是KVO的回调要被调用,属性必须是通过KVC的方法来修改的,如果是调用类的其他方法来修改属性,这个观察者是不会得到通知的。
参考文章
IOS中KVO模式的解析与应用
iOS:KVO的概述与使用
通知和kvo分别在什么时候使用
还有两篇我觉得加深理解KVO的好文章:
iOS 中KVC、KVO、NSNotification、delegate 总结及区别
iOS下KVO使用过程中的陷阱