[디자인패턴] OBSERVERPATTEN(옵저버패턴)
안녕하세요. 예지우랑입니다.
디자인패턴중 제일 처음에 나오는 observer(옵저버)패턴에 대해 알아보겠습니다.
(※ 이 포스팅의 내용과 예제는 한빛미디어의 HEAD FIRST DESIGN PATTERNS를 참조하여 작성하였습니다.)
1. OBSERVERPATTEN
-정의 : 한 객체의 상태가 바뀌면 그 객체에 의존하는 다른 객체들테 연락이 가고 자동으로 내용이 갱신되는 방식으로
일대다(一對多 , one to many)의존성을 정의합니다.
-장점 : 주제 객체의 상태변화를 자동으로 알려주어 의존하는 다른객체를 동적으로 변화시킬 수 있다.
2.예제
사실 옵저버 패턴은 매우 간단합니다.
위의 정의가 전부입니다.
하지만 좀더 정확히 하기위해서 가벼운 예제를 들어보기로 합시다.
여러분이 농장 주인이라고합시다.
여러부는 농장의 동물들에게 밥을 주는대 매번 밥을 주러 사육장을 돌아다니기 귀찮아서
사육장에 스피커를 연결해놓고 급식장에 밥을 준비한 후 벨을 울려서 동물들에게 밥이 나왔다는것을 알려줍니다.
그림 1. 처럼 말이죠
그림1. 주제와 옵저버
그림1.처럼 여러분이 급식송 밥을주면 각 사육장으로 알려주게 됩니다.
그러면 동물들은 밥을 먹으러 오게 되겠죠??
여기서 여러분이 '주제'객체이고 각 동물들은 '옵저버 그룹'에 속해있는 '옵저버 객체'가 됩니다.
여러분(주제)가 벨을 울리는 행위를 '주제'객체가 '옵저버'객체에게 상태가 업되이트 되었다는것을 알려주는 것이죠.
그런데 사업 확장을 위하여 어제부터 키우기 시작한 오리는 옵저버 그룹이아니어서(사육장에 벨이 없어서) 밥을 못먹는
구조네요.
불쌍항 오리를 위하여 그림2.처럼 오리를 옵저버 그룹에 등록시켰습니다.
그림 2. 새로운 객체가 옵저버에 추가됨.
오리를 옵저버그룸에 등록(register)하여 이제 오리도 벨소리를 듣고 밥을 먹으러 올 수 있게 되었습니다.
농장을 운영하다보니 우리의 개들이 밥을 너무 많이먹고 운동이 부족하여 돼지가 되었습니다.
돼지는 아직 사육할 계획이 없는데, 돼지가 있으니 난감한 여러분은 각종 성인병의 원이이되는 비만을 치료하기 위하여
개에게 당분간 먹이를 주지 않기로합니다.
그림3. 처럼 말이죠
그림 3. 옵저버 그룹에서 특정 객체를 제거.
여러분은 개 사육장에 연결된 벨에 전선을 끊어버렸습니다.(remove)이로 인해서
이제 개들은 다시 옵저버 그룹에 등록(register)되기 전까지는 밥을 못먹게 되었네요.
이렇게 옵저버 패턴을 이용하여 여러분은 농장의 동물들의 체중을 쉽게 조절하면서
농장을 운영할 수 있습니다.
위의 예에서 알 수 있는 옵저버 패턴의 특징은
각 객체는 언제든 주제객체에 의존하는 옵저버 그룹에 가입하거나 탈퇴할 수 있다. 라는 것입니다.
개념 정리가 어느정도 되셨나요?
이제는 실제로 적용하는 코드르 보도록 합시다.
여러분은 프로젝트를 맡았습니다.
프로젝트의 내용은 기상청에서 업데이트되는 기상정보중 기온, 습도, 기압을 이용하여
사용자에게 실시간으로 알려주는 프로그램을 만드는 것입니다.
다행이 한국 기상청은 기상정보를 국민 모두가 볼수있게 공개하고 있으며,
개발자들이 쉽게 쓸 수 있도록 API도 제공하고 있네요.
그림4.는 기상청이 제공하는 API에 있는 날씨정보 객체입니다.
그림4. 기상청이 제공해주는 API중 날씨정보 객체
getTemperature 메소드는 온도를,
getHumidity 메소드는 습도를,
getPressure 메소드는 기압을 각각 기상청에서 가저오는 메소드입니다.
measurementsChanged는 관측값이 갱신(그림엔 오타가있네요) 될때마다 알려주는 메소드입니다.
아래 코드1.에서 코드를 보도롭합시다.
public class WeatherData {
//인스턴스 변수자리
public void measurementsChanged(){
//이미 구현되어있는 WeatherData의 getter method를 이용하여 최신측정값을 가져옴
float temp = getTemperature();
float humidity = getHumidity();
gloat pressure = getPressure();
currentContitionsDisplay.update(temp,humidity,presure);
statisticsDisplay.update(temp,humidity,pressure);
forecastDisplay.update(temp, humidity, pressure);
}
//기타메소드
} |
코드1. WhatherData 클래스
위의 코드에서 measurementChanged() 메소드를 구현해 봤습니다.
각 정보 업데이트 메소드를 이용해서 변수에 정보를 담은 후
currentContitionsDisplay.update(temp,humidity,presure);
statisticsDisplay.update(temp,humidity,pressure);
forecastDisplay.update(temp, humidity, pressure);
위의 세 메소드를 이용하여 우리가 기상정보를 보여줄 현재상태, 평균/최저/최고치 , 기상예보를 표시해주는
프로그램에(객체에) 정보를 보내주고 있습니다.
이쯤되면 눈치를 체시겠지만 우리는 기상청에서 제공해주는 API를 이용하여
현재 기상상태 , 평균/최저/최고치 , 기상예보 이렇게 세가지 정보를 유저들에게 제공해줄 생각입니다.
좀더 이해를 돕기위해 그림5.의 클래스 다이어그램을 보면서 살펴봅시다.
그림5. 옵저버패턴의 클래스다이어그램
먼저 Subject 부터 봅시다.
Subject는 주제를 나타내는 인터페이스입니다.
객체를 옵저버로 등록하거나 옵저버 목록에서 탈퇴시키고 싶을때는 이 인터페이스에 있는 메소드를 사용하면됩니다.
그다음은 Observer 인터페이스입니다.
Observer가 될 가능성이 있는 객체는 반드시 Oberver인터페이스를 구현해야 합니다.
이 인터페이스에는 주제의 상태가 바뀌었을때 호출되는 update()메소드밖에 없습니다.
각 주제마다 여러게의 Oberver가 있을 수 있습니다.
ConcreteSubject는 주제 역하을 하는 구상 클래스입니다.
항상 Subject인터페이스를 구현 해야하면
주제 클래스에서는 등록 및 해지를 위함 메소드 웨에, 상태가 바뀔때마다 모든 옵저버들에게 연락을 하기위한
notifyObservers()메소드를 구현해야합니다.
또, 주제 클래스에는 상태를 설정하고 알아내기 위한 세터/게터 메소드가 있을 수도있습니다.
ConcreteObserver는 Observer인터페이스를 구현해야만 옵저버 클래스가 될 수 있습니다.
각 옵저버는 특정 주제 객체에 등록을 해서 연락을 받을 수 있습니다.
※의존성이란?
의존성이랑 한객체 또는 로직이 다른 객체의 상태에 따라 영항을 받는것을 이야기합니다.
옵저버패턴에서는 옵저버들이 주제 객체의 상태에따라 옵저버의 상태가 변하게되므로
옵저번느 주제에대해 의존성을 가진다 라고 말합니다.
디자인원칙 |
서롤 상호작용을 하는 객체 사이에서는 가능하면 느슨하게 결합을 하는 디자인을 사용해야합니다. 느슨하게 결합하는 디자인을 사용하면 변경사항이 생겨도 무난히 처리할 수 있는 유연한 객체지향 시스템을 구축할 수 있습니다. 객체 사이의 상호 의존성을 최소화 할 수 있기 때문이죠 |
옵저버 패턴의 주제와 옵저버는 느슨한결합(Loose Coupling)을 하고 있습니다. 옵저버 패턴이 느슨한 결합이라는 근거와 장점을 알아보고 갑시다. 근거1. 주제가 옵저베에 대해서 아는것은 옵저버가 특정 인터페이스를 구현한다는것 뿐입니다. 다른걸 알필요도 없고 알고싶지도 않습니다. 장점1. 옵저버는 언제든 새로 추가할 수 있습니다. 주제는 옵저버 인터페이스를 구현하는 객체에만 의존하기 때문에 언제든 새로운 옵저버를 추가할 수 있죠. 장점2. 주제와 옵저버는 독립적으로 사용할 수 있습니다. 어차피 서로에 대해 아는것은 특정 인터페이스를 구현할 수 있다는 것 뿐이고 그외에는 서로 관심도 없습니다. 옵저버가 새로운 행동을 하는것이 주제와의 관계에서 문제가 일어날거같다면 그저 주제의 목록에서 삭제해 버리면 그만입니다. 장점3. 주제나 옵저버가 바뀌더라도 서로에게 영향을 미치지 않습니다. 서로 인터페이스를 구현한다는 조건만 만족되면 어떻게 바꿔도 문제가 생기지 않습니다. |
이제 최종적으로 우리의 시스템을 구현하기 위한 클래스 다이어그램인 그림6.을 봅시다.
그림6. 우리가 구현하게될 기상정보시스템의 최종 클래스 다이어그램. (클릭하면커집니다.)
하나씩 설명을 해봅시다.
Subject는 주제 인터페이스입니다. 우리가 지금까지해온것과 동일합니다.
Obser인터페이스는 주제 객체에서 옵저버에게 생신된 정보를 전달할 수 있는 방법을 정의하게 만듭니다.
DisplayElement인터페이스는 우리의 기상 정보를 화면에 표시해주는 방법을 정의하게하는 인터페이스입니다.
WeatherData는 기상청에서 제공해주는 객체입니다. Subject를 Implement하여 우리의 시스템에서 새로 갱신된 날씨데이터를
가져다 줍니다.
CurrentConditions는 현재 측정된값을 보여줍니다.
StatisticsDisplay는 평균/최고/최저값을 보여줍니다.
ForecastDisplay는 기상 예보를 표시해줍니다.
ThirdPartyDisplay는 위의 세가지 이외에도 Obsever와 DisplayElement만 구현하면 옵저버객체가되며 정보를 표시해줄 수 있다는것을
알려주는것입니다.
이제 구현을 해봅시다.
public interface Subject { public void registerObserver(Observer o); //옵저버를 등록
public void removeObserver(Observer o); // 옵저버를 제거
public void notifyObservers(); // 주 객체의 상태 변경시 모든 옵저버에게 알려주는 메소드
} |
코드2. Subject Interface
public interface Observer { public void update(float temp , float humidity, float pressure);
/*
* 옵저버 인터페이스는 모든 옵저버 클래스에서 구현해야합니다.
* 따라서 모든 옵저버는 update메소드를
* 구현해야합니다.
*/
} |
코드3. Observer Interface
public interface DisplayElement { public void display();
/*
* display화면을 출력해야할 경우 호출해야되는 메소드.
*/
} |
코드4. DisplayElement InterFace
import java.util.ArrayList; public class WeatherData implements Subject {
private ArrayList observers; //ArrayList자료구조를 사용합니다. ArrayList를 import합시다.
private float temperature;
private float humidity;
private float pressure;
public WeatherData(){
observers = new ArrayList(); //생성자에서 observers를 생성 해 줍니다.
}
public void registerObserver(Observer o){ //옵저버가 등록을 하면 목록 맨뒤에 추가해줍니다.
observers.add(o);
}
public void removeObserver(Observer o){ //옵저버가 탈퇴를 요청하면 목록에서 제거해줍니다.
int idx = observers.indexOf(o);
if(idx >=0 ){
observers.remove(idx);
}
}
public void notifyObservers(){
for(int i = 0 ; i < observers.size(); i++){
Observer observer = (Observer)observers.get(i);
observer.update(temperature, humidity, pressure);
}
}
public void measurementsChanged(){
notifyObservers();
}
public void setMeasurements(float temperature , float humidity, float pressure){
/*
* 기상청에서는 날씨정보를 알려주는 API를 제공합니다.
* 그러나 여기서 적용시키자면 배보다 배꼽이 커지게되어 이 메소드를 통하여
* 확인해보도록하겠습니다.
* http://www.kma.go.kr/weather/lifenindustry/sevice_website.jsp
* 진짜 날씨정보를 제공해주는 프로그램을 만드시고싶다면
* 위의 페이지를 참조하여주세요.
*/
this.temperature = temperature ;
this.humidity = humidity;
this.pressure = pressure;
measurementsChanged();
}
//기타메소드
} |
코드5. 실제 구현한 WeatherData Class
public class CurrentConditionsDisplay implements Observer, DisplayElement {
/*
* WeatherData객체로부터 변경사항을 받이귀아혀 Observer객체를 구현합니다.
* API구조상 모든 디스플레이 항목에서 DisplayElement를 구현하기로 했기 때문에
* DisplayElement인터페이스도 구현합니다.
*/
private float temperature;
private float humidity;
private Subject weatherData;
public CurrentConditionsDisplay(Subject weatherData){
/*
* 생성자에 weatherData라는 주제객체가 전달되며 , 그 객체를 써서 디스플레이를 옵저버로 등록합니다.
*/
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
public void update(float temperature, float humidity , float pressure){
/*update()가 호출되면 기온과 습도를 저장하고 display()를 호출합니다.*/
this.temperature = temperature;
this.humidity = humidity;
display();
}
public void display(){
System.out.println("현재 기온:" + temperature + " | 현재 습도:" + humidity);
}
} |
코드6. CurrentConditionsDisplay클래스
/*테스트용 클래스입니다. 우리가 만든 프로그램이 잘 동작하는지 알아봅시다.*/
public class WeatherStation {
public static void main(String[] args){
WeatherData weatherData = new WeatherData();
//우선 웨더 데이터 객체를 생성합니다.
CurrentConditionsDisplay currentDisplay = new CurrentConditionsDisplay(weatherData);
//현재값을 보여주는 클래스인 CurrentConditionsDisplay 객체를 생성합니다. 인자로 미리 생성한 웨더데이터 객체를 줍니다.
weatherData.setMeasurements(80, 65, 30.5f);
weatherData.setMeasurements(10, 20, 88.7f);
weatherData.setMeasurements(65, 23, 10.5f);
//세로운 기상 값이 들어간것처럼 만들어줍니다.
}
} |
코드7. 테스트를 위해서 main메소드를 만들어봤습니다. 한번 돌려봅시다.
최대/최저/평균을 구하는 staticsDisplay클래스와
기상예보를 해주는 ForecastDisplay클래스는 여러분이 직접 만들어보시면 좋을거같습니다.!!!!
3. 옵저버패턴은 실제 어디에서쓰일까?
옵저버패턴은 일종의 push방식의 알고리즘입니다.
업데이트된 정보 또는 DATA를 처리하는 알고리즘에는 크게 PUSH방식과 PULL방식이 있습니다.
물론 두가지를 결합한 형태의 방식도 존재하죠
지금 알아본 옵저버패턴은 일종의 push방식 알고리즘으로써
상태가 바뀐값을 다른 객체에 알려주면서(push하면서) 그 값을 다른 객체가 처리하고있죠
제가 생각했을때 일상에서 가장 쉽게 접할수있는 시스템중에 옵저버 패턴을 적욕할 법한 시스템은
쇼핑몰입니다.
우리가 구매 버튼을 누른 후 결제를 하고면 자동으로 우리 이메일로
결제가 완료되었습니다. 라는 안내메일이 오잖아요?
여러 구현 방법이 있을 수 있겠지만 결제모듈을 주제 객체로, 메일링 시스템을 옵저버 객체로 등록해놓고
결제완료시에 update()메소드를 호출하면 자동으로 메일링 시스템이 메일을 보내주겠죠?
4. 최종적으로 옵저버패턴의 핵심을 다시 봅시다.
OBSERVERPATTEN
-정의 : 한 객체의 상태가 바뀌면 그 객체에 의존하는 다른 객체들테 연락이 가고 자동으로 내용이 갱신되는 방식으로
일대다(一對多 , one to many)의존성을 정의합니다.
-장점 : 주제 객체의 상태변화를 자동으로 알려주어 의존하는 다른객체를 동적으로 변화시킬 수 있다
옵저버 패턴의 특징
각 객체는 언제든 주제객체에 의존하는 옵저버 그룹에 가입하거나 탈퇴할 수 있다.
※의존성이란?
의존성이랑 한객체 또는 로직이 다른 객체의 상태에 따라 영항을 받는것을 이야기합니다.
옵저버패턴에서는 옵저버들이 주제 객체의 상태에따라 옵저버의 상태가 변하게되므로
옵저번느 주제에대해 의존성을 가진다 라고 말합니다.
디자인원칙 |
서롤 상호작용을 하는 객체 사이에서는 가능하면 느슨하게 결합을 하는 디자인을 사용해야합니다. 느슨하게 결합하는 디자인을 사용하면 변경사항이 생겨도 무난히 처리할 수 있는 유연한 객체지향 시스템을 구축할 수 있습니다. 객체 사이의 상호 의존성을 최소화 할 수 있기 때문이죠 |
옵저버 패턴의 주제와 옵저버는 느슨한결합(Loose Coupling)을 하고 있습니다. 옵저버 패턴이 느슨한 결합이라는 근거와 장점을 알아보고 갑시다. 근거1. 주제가 옵저베에 대해서 아는것은 옵저버가 특정 인터페이스를 구현한다는것 뿐입니다. 다른걸 알필요도 없고 알고싶지도 않습니다. 장점1. 옵저버는 언제든 새로 추가할 수 있습니다. 주제는 옵저버 인터페이스를 구현하는 객체에만 의존하기 때문에 언제든 새로운 옵저버를 추가할 수 있죠. 장점2. 주제와 옵저버는 독립적으로 사용할 수 있습니다. 어차피 서로에 대해 아는것은 특정 인터페이스를 구현할 수 있다는 것 뿐이고 그외에는 서로 관심도 없습니다. 옵저버가 새로운 행동을 하는것이 주제와의 관계에서 문제가 일어날거같다면 그저 주제의 목록에서 삭제해 버리면 그만입니다. 장점3. 주제나 옵저버가 바뀌더라도 서로에게 영향을 미치지 않습니다. 서로 인터페이스를 구현한다는 조건만 만족되면 어떻게 바꿔도 문제가 생기지 않습니다. |
다음번엔 또 새로운 패턴으로 돌아오겠습니다. 감사합니다.