iOS/Design Pattern
Design Pattern - Observer Pattern
HaningYa
2020. 8. 9. 19:18
728x90
Observer Pattern
Observer 패턴은 하나의 Object 가 다른 Object의 변화를 관찰 할 수 있는 패턴이다.
Apple 은 이 패턴을 Swift5.1에 언어수준에서 RxSwift 와 비슷한 Publisher 와 Combine 프레임워크를 통해서 공식적으로 지원하는 패턴으로 추가하였다.
패턴은 세가지 타입을 포함한다.
- subscriber: 변경된 사항에 대해 업데이트를 받는 관찰자 object
- publisher: observable(관찰가능한) object, 변경되면 업데이트를 subscriber 에게 알려줌
- value: 객체의 변경된 값
Observer Pattern - 언제 사용해야 할까
- 다른 객체의 변경사항을 알고 싶을때
- MVC 패턴에서 주로 쓰인다.
Subscriber: view controller
Publisher: model
--> model 에 대한 변경값이 view controller 로 전달되게 해준다. (view controller 타입에 대한 내용 없이) - 어떤 view controller 인지 상관없기 때문에 여러 다른 view controller 들이 같은 model 타입에 대한 변경값을 observe 할 수 있다.
Observer Pattern - Behavior Pattern
Observer Pattern 은 다른 객체를 observing 하는 패턴 이므로 행위 패턴이다.
Observer Pattern 예제 (Combine)
import Combine
public class User {
@Published var name: String
public init(name: String) {
self.name = name
}
}
- 먼저 Combine 을 import 해준다. (@Published 어노테이션과 Publisher, Subscriber 타입을 가지고 있다.)
- 새로운 User 클래스를 만든다. (@Published 프로퍼티는 class 가 아닌 struct 나 다른 타입에는 사용할 수 없다.)
- name 변수를 선언하고 @Published 라고 표기한다. 그러면 Xcode가 자동으로 Publisher 를 만들어 준다. (@Published는 let 에 쓸수 없다. --> 당연하지)
- initializer 를 만들고 name 의 초기값을 정해준다.
예제를 실행해 보자
let user = User(name: "Ray")
let publisher = user.$name
var subscriber: AnyCancellable? = publisher.sink(){
print("User's name is \($0)")
}
user.name = "Vicki"
- 먼저 새로운 user 인 Ray 를 만든다.
- user 이름에 대한 변경사항을 전달하기 위해 user.$name 을 통해서 publisher 에 접근한다.
이것은 Published<String>.Publisher 를 리턴한다. - publisher 에 sink 를 호출 함으로써 subscriber 를 만든다.
초기값과 값이 변경될 때마다 호출될 클로져가 필요하다.
defualt 로 sink 는 AnyCancellable 타입을 리턴한다. 하지만 명시적으로 AnyCancellable? 처럼 옵셔널로 처리하여 나중에 nil 처리 할 수 있다. - 마지막으로 user name 을 Vicki 로 바꾼다.
//실행결과
User's name is Ray
//user 이름을 Vicki 로 바꾼 직후
User's name is Vicki
subscriber 를 nil 로 설정하면 더이상 publisher 로 부터 값을 update 받지 않는다.
subscriber = nil
user.name = "change name"
//값을 바꿔도 아무 출력 안됨
Observer Pattern 사용시 주의할 점
- observer pattern 을 적용하기 전 어떤 조건 (condition) 에서 어떤 변화(change)가 일어나길 바라는지 생각해라
- object 나 property 가 change 할 이유를 찾지 못할 것 같으면 var 이나 @Publisehd 말고 let으로 선언해라
- unique identifier을 예로 들자면 definition이 절대 바뀌지 않으므로 published property 로 사용하기 적절치 않다.
[Combine]
728x90