콘텐츠로 건너뛰기
Home » [cocoa] 키-밸류 옵져빙 간단 예제

[cocoa] 키-밸류 옵져빙 간단 예제

키-밸류 옵저빙(Key-Value Observing)은 코코아의 기초적인 기능 중 하나로, 특정 객체의 프로퍼티가 변경되었을 때, 다른 객체(옵저버)로 하여금 이를 알아차릴 수 있도록 메시지를 자동으로 보내주는 것이다. 특정 객체의 접근자에서 프로퍼티를 변경할 때마다 콜백을 호출하는 식으로 코드를 작성할 필요가 없고, 객체 외부에서 옵저버를 설치할 수 있기 때문에 다양한 상황에서 사용할 수 있다.

프로퍼티의 변경을 감지하고자 하는 대상은 NSObject의 서브 클래스이면서, 해당 프로퍼티는 키밸류 코딩 호환 가능하도록 정의된 프로퍼티여야 한다. 이 때, 해당 프로퍼티는 -setValue:ForKey: 를 사용하여 간접적으로 변경하지 않고 [anObject setProperty:value]와 같은 식으로 일반적인 프로퍼티 접근자를 사용하는 경우에도 작동한다.

키밸류 옵저빙은 특히 모델 객체의 속성과 UI 객체의 모양을 양방향으로 연결하여 데이터 모델의 변경이 UI에 자동으로 반영되도록 하는 코코아 바인딩을 가능하게 하는 기본 기능이 되며, 그 외에도 특정 값의 변경을 감시해야 하는 상황에서 유용하게 사용될 수 있다.

키밸류 옵저빙을 사용하는 방법은 다음과 같다.

  1. 감지하고자 하는 대상의 프로퍼티를 키밸류 코딩에 호환되도록 작성한다.
  2. 감시 대상이 되는 객체에 옵저버를 추가한다. (-addObserver:forKeyPath:options:context:)
  3. 옵저버는 감시하는 프로퍼티가 변경되면 -observeValueForKeyPath:options:context: 메시지를 받게 되니, 해당 메소드를 오버라이드해 준다.

객체에 옵저버를 설치하기

어떤 객체의 프로퍼티에 대해 옵저버를 추가하는 코드는 대략 다음과 같다.

[self addObserver:self
      forKeyPath:@"self.aNumber.intValue"
      options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld
      context: null];

option: 파라미터는 옵저버에게 어떤 정보를 전달할 것인지를 알려주는 마스크 값을 전달한다. NSKeyValueObservingOption*으로 명명되는 이 옵션들을 사용하면 된다. 위 예에서는 새로 설정된 값 및 이전 값을 전달하게 된다. context: 파라미터는 추가적인 정보를 함께 전달하고 싶을 때 사용하면 된다.

옵저버가 변경을 감지할 때

옵저버가 되는 객체는 감시 대상 프로퍼티가 변경될 때마다 observeValueForKeyPath:ofObject:change:context:라는 메시지를 받게 된다. 여러 객체를 하나의 옵저버가 감시할 수 있으므로 키패스와 객체 참조를 통해서 특정한 대상의 프로퍼티마다 적절한 동작을 하도록 구현할 수 있다. 변경 사항은 사전의 형태로 전달되어 이전값 및 새로운 값을 구분할 수 있다. 컨텍스트 정보는 옵저버를 등록할 때 전달해놓은 정보를 다시 돌려받게 된다.

- (void)observeValueForKeyPath:(NSString *)keypath
                   ofObject:(NSObject *)object
                   change:(NSDictionary *)change
                   context:(void *)context
{
    if ([keypath isEqualToString:@"self.number.intValue"]) {
        NSLog(@"변경이 감지되었습니다.");
    }
}

옵저버 해제하기

감시 대상이 되는 객체는 옵저버에 대한 참조를 하나 가지게 된다. 이는 컴파일러에 의해 추적되지 않기 때문에 ARC의 적용을 받지 못한다. 따라서 감시 대상이 되는 객체가 파괴될 때에는 옵저버를 제거하는 코드를 반드시 호출해주어야 한다. 감지 대상은 옵저버를 직접적으로 참조하는 속성을 가지고 있지 않기 때문에, 키패스에 의존해서 옵저버를 제거한다. 이 때 호출하는 메소드는 -removeObserverForKeyPath:context: 이다.