以钓鱼为例,要实现完整的KVO,需要以下条件:
- 钓鱼者,行为就是钓鱼,变化的是钓到的鱼数。
- 观察者,行为是观察钓鱼者的行为,每钓到一条鱼就报告一次。
- 显示,接收观察者的报告,将最新消息展示出来。
相对应的,我们可以定义三个对象:
- Fisherman,属性是鱼数,方法是钓鱼,假设每3秒钓到一条鱼,钓够20条鱼回家。
- DLObserver,注册协议报告鱼数,属性是被观察的钓鱼者和协议。
- ViewController,服从观察者的协议,属性是钓鱼者,观察者以及显示器label,在协议方法里,修改钓鱼者钓到的鱼数。
下面,直接上代码:
钓鱼者(Fisherman)
接口文件(.h文件)
/*
钓鱼者:每3秒钓一条鱼,当钓到20条时,回家
*/
@interface Fisherman : NSObject
@property (nonatomic, assign) NSInteger numbers;//钓到的鱼数
- (void)fishNumbers;//钓鱼的行为
@end
实现文件(.m文件)
@implementation Fisherman {
NSTimer *timer;
}
- (instancetype)init {
if (self = [super init]) {
timer = [NSTimer scheduledTimerWithTimeInterval:3 target:self selector:@selector(fishNumbers) userInfo:nil repeats:YES];
}
return self;
}
- (void)fishNumbers {
NSInteger numbers = self.numbers;
if (numbers == 20) {
[timer invalidate];
return;
}
[self setNumbers:++numbers];
NSLog(@"被观察者钓到的鱼数:%ld", self.numbers);
}
@end
观察者(DLObserver)
接口文件(.h文件)
/*
观察者:观察被观察者的行为,并报告
*/
//观察者报告的协议
@protocol DLServerReportDelegate<NSObject>
//报告被观察者钓到的鱼数
- (void)reportFishNumbers:(NSInteger)numbers;
@end
@interface DLObserver : NSObject
//被观察的钓鱼者对象
@property (nonatomic, strong) Fisherman *fisher;
//delegate
@property (nonatomic, assign) id<DLServerReportDelegate>delegate;
//初始化,并记录被观察的钓鱼者
- (instancetype)initWithFisher:(Fisherman *)fisher;
@end
实现文件(.m文件)
@implementation DLObserver
//初始化
- (instancetype)initWithFisher:(Fisherman *)fisher {
if (self = [super init]) {
self.fisher = fisher;
//给钓鱼者添加观察者
[self.fisher addObserver:self forKeyPath:@"numbers" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:nil];
}
return self;
}
//观察者方法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
//观察者通过协议报告
if ([keyPath isEqualToString:@"numbers"]) {
//最新钓到的鱼数
NSInteger numbers = [change[@"new"] integerValue];
if ([self.delegate respondsToSelector:@selector(reportFishNumbers:)]) {
[self.delegate reportFishNumbers:numbers];
}
}
}
//移除观察者
- (void)dealloc {
[self.fisher removeObserver:self forKeyPath:@"numbers"];
}
@end
显示(ViewController)
实现文件(.m文件)
@interface ViewController ()<DLServerReportDelegate>
{
Fisherman *fisher;//钓鱼者
DLObserver *observer;//观察者
}
@property (weak, nonatomic) IBOutlet UILabel *numberLabel;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
fisher = [[Fisherman alloc] init];
}
//开始观察钓鱼者
- (IBAction)startObserve:(id)sender {
if (!observer) {
observer = [[DLObserver alloc] initWithFisher:fisher];
observer.delegate = self;
}
}
//更新报告过来的鱼数
- (void)reportFishNumbers:(NSInteger)numbers {
_numberLabel.text = [NSString stringWithFormat:@"%ld条", numbers];
}
@end
效果图如下:
效果图
够详细了吧,那就不用再贴出demo地址了。