옵저버 패턴

옵저버(Observer) 패턴은 한 객체의 상태가 변경되면, 이 객체에 의존하는 다른 객체들에게 자동으로 알림이 전달되도록 하는 디자인 패턴입니다.
주로 이벤트 기반 시스템이나 데이터의 변경 사항을 실시간으로 반영해야 하는 경우에 사용됩니다.

여기서 한 객체는 주제(Subject)라고 하며, 이 객체에 의존하는 다른 객체를 옵저버(Observers)라고 합니다.

느슨한 결합을 가능하게 하며, 주제와 옵저버간의 의존성을 최소화 하고, 인터페이스를 통해 소통하기 때문에 객체 간의 독립성을 유지할 수 있습니다.
주제와 옵저버를 수정, 교체하거나 새로운 옵저버를 추가, 제거해도 주제의 코드에 영향을 미치지 않는 독립적인 구조이므로, 유지보수를 용이하게 한다는 장점이 있습니다.

객체 간의 1:N 의존성을 정의하는 패턴이라고 할 수 있습니다.
예를 들어 게임 상태 변경 알림과 관련해서 적용해볼 수 있습니다.
게임에서 플레이어가 데미지를 입거나 회복되면 체력 상태가 변경되고, 이를 기반으로 체력바 표시 UI가 업데이트 되고, 데미지를 입거나 체력이 회복되는 사운드 효과가 재생되고, 특정 경우 게임 오버 처리 등이 실행될 수 있습니다.

옵저버가 많아질수록 알림 비용이 증가해 성능에 영향을 줄 수 있습니다.

순환 참조가 발생하지 않도록 주의해야합니다.

옵저버 패턴 예시

#include <iostream>
#include <vector>
#include <memory>
#include <string>
#include <algorithm> // std::remove_if

using namespace std;

// 추상 옵저버 (Observer): 체력 변경 이벤트를 관찰하는 인터페이스
class IObserver {
public:
    virtual ~IObserver() = default;
    virtual void Update(int health) = 0; // 상태 변경 시 호출
};

// 주제 (Subject): 체력을 관리하고 옵저버들에게 상태 변화를 알림
class Player {
private:
    vector<shared_ptr<IObserver>> observers; // 옵저버 목록
    int health;                              // 플레이어 체력

public:
    Player(int initialHealth) : health(initialHealth) {}

    // 옵저버 등록
    void Attach(const shared_ptr<IObserver>& observer) {
        observers.push_back(observer);
    }

    // 옵저버 제거
    void Detach(const shared_ptr<IObserver>& observer) {
        observers.erase(remove(observers.begin(), observers.end(), observer), observers.end());
    }

    // 상태 변경 시 모든 옵저버에게 알림
    void Notify() {
        for (const auto& observer : observers) {
            observer->Update(health);
        }
    }

    // 체력 감소
    void TakeDamage(int damage) {
        health -= damage;
        if (health < 0) health = 0; // 체력이 음수로 내려가지 않도록 제한
        cout << "Player took " << damage << " damage. Current health: " << health << endl;
        Notify(); // 옵저버들에게 상태 변경 알림
    }

    // 체력 회복
    void Heal(int amount) {
        health += amount;
        cout << "Player healed " << amount << ". Current health: " << health << endl;
        Notify(); // 옵저버들에게 상태 변경 알림
    }
};

// 구체 옵저버 (Concrete Observers)

// UI 업데이트 옵저버
class HealthBar : public IObserver {
public:
    void Update(int health) override {
        cout << "[UI] Health bar updated: " << health << endl;
    }
};

// 사운드 효과 옵저버
class SoundEffect : public IObserver {
public:
    void Update(int health) override {
        if (health > 0) {
            cout << "[Sound] Damage sound played." << endl;
        } else {
            cout << "[Sound] Game over sound played." << endl;
        }
    }
};

// 게임 오버 처리 옵저버
class GameOverHandler : public IObserver {
public:
    void Update(int health) override {
        if (health <= 0) {
            cout << "[GameOver] Player is dead. Game Over!" << endl;
        }
    }
};

int main() {
    // 1. 주제(Subject) 생성
    auto player = make_shared<Player>(100); // 초기 체력 100

    // 2. 옵저버(Observers) 생성
    auto healthBar = make_shared<HealthBar>();
    auto soundEffect = make_shared<SoundEffect>();
    auto gameOverHandler = make_shared<GameOverHandler>();

    // 3. 옵저버 등록
    player->Attach(healthBar);
    player->Attach(soundEffect);
    player->Attach(gameOverHandler);

    // 4. 플레이어 체력 변화
    player->TakeDamage(30); // 체력 100 -> 70
    player->TakeDamage(50); // 체력 70 -> 20
    player->TakeDamage(25); // 체력 20 -> 0, 게임 오버

    // 5. 플레이어 체력 회복
    player->Heal(50); // 체력 회복 (옵저버가 상태를 업데이트)

    return 0;
}

DesignPattern 카테고리 내 다른 글 보러가기

댓글남기기