iOS KVO观察对象属性值的变化

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 

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 *)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使用过程中的陷阱

你可能感兴趣的:(iOS KVO观察对象属性值的变化)