Key-Value Observing

原创 2016年05月30日 23:54:25

1.简介

在ios里,可以添加观察者模式,来实现某个property更改后,通知指定的类。
然后到observeValueForKeyPath:ofObject:change:context:提供处理
1 [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

2.demo示例1:
//
//  KVCTableViewController.m
//  KVC
//
//  Created by tinghou on 16/5/30.
//

#import <UIKit/UIKit.h>

@interface KVCTableViewCell : UITableViewCell
- (id)initWithReuseIdentifier:(NSString*)identifier;
@property (nonatomic, readwrite, strong) id object;
@property (nonatomic, readwrite, copy) NSString *property;
@end

//
//  KVCTableViewController.m
//  KVC
//
//  Created by tinghou on 16/5/30.
//

#import "KVCTableViewCell.h"

@implementation KVCTableViewCell

- (BOOL)isReady {
  return (self.object && [self.property length] > 0);
}

- (void)update {
  self.textLabel.text = self.isReady ?
  [[self.object valueForKeyPath:self.property] description]
  : @"";
}

- (id)initWithReuseIdentifier:(NSString *)identifier {
  return [super initWithStyle:UITableViewCellStyleDefault
              reuseIdentifier:identifier];
}

- (void)removeObservation {
  if (self.isReady) {
    [self.object removeObserver:self
                     forKeyPath:self.property];
  }
}
#pragma mark 添加观察者
- (void)addObservation {
  if (self.isReady) {
    [self.object addObserver:self forKeyPath:self.property
                     options:0 
                     context:(void*)self];
  }
}
#pragma mark 一旦监听到改变就会调用
- (void)observeValueForKeyPath:(NSString *)keyPath 
                      ofObject:(id)object 
                        change:(NSDictionary *)change 
                       context:(void *)context {
  if ((__bridge id)context == self) {
      [self update];
  }
  else {
    [super observeValueForKeyPath:keyPath ofObject:object 
                           change:change context:context];
  }
}

- (void)dealloc {
  if (_object && [_property length] > 0) {
    [_object removeObserver:self
                 forKeyPath:_property
                    context:(void *)self];
  }
}

- (void)setObject:(id)anObject {
  [self removeObservation];
  _object = anObject;
  [self addObservation];
  [self update];
}

- (void)setProperty:(NSString *)aProperty {
  [self removeObservation];
  _property = aProperty;
  [self addObservation];
  [self update];
}
@end

//
//  KVCTableViewController.m
//  KVC
//
//  Created by tinghou on 16/5/30.
//

#import "KVCTableViewController.h"
#import "KVCTableViewCell.h"
#import "RNTimer.h"

@interface KVCTableViewController ()
@property (readwrite, strong) RNTimer *timer;
@property (readwrite, strong) NSDate *now;
@end

@implementation KVCTableViewController

- (void)updateNow {
  self.now = [NSDate date];
}

- (void)viewDidLoad {
  [self updateNow];

  __weak id weakSelf = self;
  self.timer =
      [RNTimer repeatingTimerWithTimeInterval:1
                                        block:^{
                                          [weakSelf updateNow];
                                        }];
}

- (void)viewDidUnload {
  self.timer = nil;
  self.now = nil;
}

- (NSInteger)tableView:(UITableView *)tableView
 numberOfRowsInSection:(NSInteger)section {
  return 100;
}

- (UITableViewCell *)tableView:(UITableView *)tableView
         cellForRowAtIndexPath:(NSIndexPath *)indexPath {
  
  static NSString *CellIdentifier = @"KVCTableViewCell";
  
  KVCTableViewCell *cell = [tableView
             dequeueReusableCellWithIdentifier:CellIdentifier];
  
  if (cell == nil) {
    cell = [[KVCTableViewCell alloc]
            initWithReuseIdentifier:CellIdentifier];
    [cell setProperty:@"now"];
      
    [cell setObject:self];
  }
    
  return cell;
}

@end
项目目录:

版权声明:本文为博主原创文章,未经博主允许不得转载。

Key-Value Observing(KVO)详解

Key-Value Observing机制 知识点介绍 Key-Value Observing (简写为KVO):当指定的对象的属性被修改了,允许对象接受到通知的机制。每次指定的被观察对象...

iOS---KVO(Key Value Observing) 观察者模式之解析与应用

一、概述 KVO(Key Value Observing) 观察者设计模式。通过KVO这种机制对象可以通过它得到其他对象的某个属性的变更通知。KVO可以让视图对象经过控制器观察模型对...

KVO(Key-Value Observing)

参考文档: 《Cocoa Programming for Mac OS X 4》 上一篇讲解KVC中,假如sliderNumber的值不是被滑动条改变而是被其他对象改变,那么滑动条如何知道s...

KVO编程指南,Key-Value Observing Programming Guide翻译 - iOS

本文介绍Key-Value Observing,即KVO键值观察编程指南。包括怎么注册,移除观察者,使用场景。和键值编程相关的兼容性,手动通知和自动通知;一对一,一对多关系...

Key-Value Observing

Key-Value Observing 快速入门 假如需要掌握Key-Value Observing机制,那么需要阅读本文应该有帮助。本文提供了Key-Value Observing的一个概...

Key-Value Observing机制

Key-Value Observing机制 知识点介绍 Key-Value Observing (简写为KVO):当指定的对象的属性被修改了,允许对象接受到通知的机制。每次指定的被观察对象的属性被...

Objective-C语言--Key-Value Observing Programming Guide

Key-Value Observing Programming Guide

KVO(key value observing)的使用

简介: 上篇我们讲到了KVC,这篇我们学习KVO,全名为:Key Value Observing,直译为:基于键值的观察者。 那它有什么用呢?KVO主要用于视图交互方面,比如界面的某些数据...
  • sinyran
  • sinyran
  • 2012年10月16日 13:58
  • 456

Key-Value Observing (Brief Intro)

Key-Value Observing Key-value observing (KVO) is a mechanism that enables objects to be notifie...

ios开发kvo模式基础-Key-Value Observing 快速入门

假如需要掌握Key-Value Observing机制,那么需要阅读本文应该有帮助。本文提供了Key-Value Observing的一个概述,并且解释为什么需要用它,什么时候用它。在阅读本文之前需要...
  • niitlcj
  • niitlcj
  • 2012年02月28日 14:25
  • 454
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Key-Value Observing
举报原因:
原因补充:

(最多只允许输入30个字)