首页 热点资讯 义务教育 高等教育 出国留学 考研考公
您的当前位置:首页正文

iOS用观察者和协议实现KVO

2024-12-20 来源:化拓教育网

以钓鱼为例,要实现完整的KVO,需要以下条件:

  1. 钓鱼者,行为就是钓鱼,变化的是钓到的鱼数。
  2. 观察者,行为是观察钓鱼者的行为,每钓到一条鱼就报告一次。
  3. 显示,接收观察者的报告,将最新消息展示出来。

相对应的,我们可以定义三个对象:

  1. Fisherman,属性是鱼数,方法是钓鱼,假设每3秒钓到一条鱼,钓够20条鱼回家。
  2. DLObserver,注册协议报告鱼数,属性是被观察的钓鱼者和协议。
  3. 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地址了。

显示全文