수색…


비고

여기에 이미지 설명을 입력하십시오.

의도는 무엇입니까?

  • 우려의 분리 원칙을 채택하십시오.
  • 피사체와 관찰자를 분리하십시오.
  • 여러 명의 관찰자가 반응하여 하나의 주제를 바꿀 수있게하십시오.

구조는 무엇입니까?

  • 제목은 등록, 등록 취소, 통지 방법을 제공합니다.
  • 옵서버는 업데이트 방법을 제공합니다.

옵저버 / 자바

관찰자 패턴은 클래스의 사용자가이 클래스가 데이터를 처리 할 때 발생하는 이벤트에 가입하고 이러한 이벤트가 발생할 때 통지를받습니다. 다음 예제에서는 처리 클래스와 관찰자 클래스를 만듭니다.이 클래스는 5 문자보다 긴 단어를 찾으면 구문을 처리하는 동안 통보됩니다.

LongWordsObserver 인터페이스는 옵저버를 정의합니다. 관찰자를 이벤트에 등록하려면이 인터페이스를 구현하십시오.

// an observe that can be registered and receive notifications
public interface LongWordsObserver {
    void notify(WordEvent event);
}

WordEvent 클래스는 특정 이벤트가 발생하면 관찰자 클래스로 전송되는 이벤트입니다 (이 경우 긴 단어가 발견됨)

// An event class which contains the long word that was found
public class WordEvent {

    private String word;

    public WordEvent(String word) {
        this.word = word;
    }

    public String getWord() {
        return word;
    }
}

PhraseProcessor 클래스는 주어진 구문을 처리하는 클래스입니다. addObserver 메소드를 사용하여 옵저버를 등록 할 수 있습니다. 긴 단어가 발견되면이 관찰자는 WordEvent 클래스의 인스턴스를 사용하여 호출됩니다.

import java.util.ArrayList;
import java.util.List;

public class PhraseProcessor {

    // the list of observers
    private List<LongWordsObserver> observers = new ArrayList<>();

    // register an observer
    public void addObserver(LongWordsObserver observer) {
        observers.add(observer);
    }

    // inform all the observers that a long word was found
    private void informObservers(String word) {
        observers.forEach(o -> o.notify(new WordEvent(word)));
    }

    // the main method - process a phrase and look for long words. If such are found,
    // notify all the observers
    public void process(String phrase) {
        for (String word : phrase.split(" ")) {
            if (word.length() > 5) {
                informObservers(word);
            }
        }
    }
}

LongWordsExample 클래스는 관찰자를 등록하고 process 메서드를 호출하며 긴 단어가 발견되면 경고를받는 방법을 보여줍니다.

import java.util.ArrayList;
import java.util.List;

public class LongWordsExample {

    public static void main(String[] args) {

        // create a list of words to be filled when long words were found
        List<String> longWords = new ArrayList<>();

        // create the PhraseProcessor class
        PhraseProcessor processor = new PhraseProcessor();

        // register an observer and specify what it should do when it receives events,
        // namely to append long words in the longwords list
        processor.addObserver(event -> longWords.add(event.getWord()));

        // call the process method 
        processor.process("Lorem ipsum dolor sit amet, consectetuer adipiscing elit");

        // show the list of long words after the processing is done
        System.out.println(String.join(", ", longWords));
        // consectetuer, adipiscing
    }
}

IObservable 및 IObserver를 사용하는 옵저버 (C #)

IObserver<T>IObservable<T> 인터페이스는 .NET에서 관찰자 패턴을 구현하는 데 사용될 수 있습니다.

  • IObservable<T> 인터페이스는 알림을 보내는 클래스를 나타냅니다.
  • IObserver<T> 인터페이스는이를 수신하는 클래스를 나타냅니다.
public class Stock {
  private string Symbol { get; set; }
  private decimal Price { get; set; }
}

public class Investor : IObserver<Stock> {
  public IDisposable unsubscriber;
  public virtual void Subscribe(IObservable<Stock> provider) {
    if(provider != null) {
      unsubscriber = provider.Subscribe(this);
    }
  }
  public virtual void OnCompleted() {
    unsubscriber.Dispose();
  }
  public virtual void OnError(Exception e) {
  }
  public virtual void OnNext(Stock stock) {
  }
}

public class StockTrader : IObservable<Stock> {
  public StockTrader() {
    observers = new List<IObserver<Stock>>();
  }
  private IList<IObserver<Stock>> observers;
  public IDisposable Subscribe(IObserver<Stock> observer) {
    if(!observers.Contains(observer)) {
      observers.Add(observer);
    }
    return new Unsubscriber(observers, observer);
  }
  public class Unsubscriber : IDisposable {
    private IList<IObserver<Stock>> _observers;
    private IObserver<Stock> _observer;

    public Unsubscriber(IList<IObserver<Stock>> observers, IObserver<Stock> observer) {
      _observers = observers;
      _observer = observer;
    }

    public void Dispose() {
      Dispose(true);
    }
    private bool _disposed = false;
    protected virtual void Dispose(bool disposing) {
      if(_disposed) {
        return;
      }
      if(disposing) {
        if(_observer != null && _observers.Contains(_observer)) {
          _observers.Remove(_observer);
        }
      }
      _disposed = true;
    }
  }
  public void Trade(Stock stock) {
    foreach(var observer in observers) {
      if(stock== null) {
        observer.OnError(new ArgumentNullException());
      }
      observer.OnNext(stock);
    }
  }
  public void End() {
    foreach(var observer in observers.ToArray()) {
      observer.OnCompleted();
    }
    observers.Clear();
  }
}

용법

...
var provider = new StockTrader();
var i1 = new Investor();
i1.Subscribe(provider);
var i2 = new Investor();
i2.Subscribe(provider);

provider.Trade(new Stock());
provider.Trade(new Stock());
provider.Trade(null);
provider.End();
...

REF : .NET에서 디자인 패턴 및 관행 : Observer 패턴



Modified text is an extract of the original Stack Overflow Documentation
아래 라이선스 CC BY-SA 3.0
와 제휴하지 않음 Stack Overflow