Примечание.
Для доступа к этой странице требуется авторизация. Вы можете попробовать войти или изменить каталоги.
Для доступа к этой странице требуется авторизация. Вы можете попробовать изменить каталоги.
Шаблон проектирования наблюдателя требует разделения между наблюдателем, который регистрирует уведомления и поставщик, который отслеживает данные и отправляет уведомления одному или нескольким наблюдателям. В этом разделе описывается создание наблюдателя. Связанная тема, Как: Реализовать поставщика, обсуждает, как создать поставщика.
Создание наблюдателя
Определите наблюдателя, который является типом, реализующим интерфейс System.IObserver<T>. Например, следующий код определяет тип с именем
TemperatureReporter, который представляет собой созданную реализацию System.IObserver<T> с аргументом универсального типаTemperature.public class TemperatureReporter : IObserver<Temperature>Public Class TemperatureReporter : Implements IObserver(Of Temperature)Если наблюдатель может перестать получать уведомления, прежде чем поставщик вызывает свою реализацию IObserver<T>.OnCompleted, определите частную переменную, которая будет содержать реализацию IDisposable, возвращаемую методом IObservable<T>.Subscribe поставщика. Также следует определить метод подписки, который вызывает метод Subscribe поставщика и сохраняет возвращенный объект IDisposable. Например, следующий код определяет частную переменную с именем
unsubscriberи определяет методSubscribe, который вызывает метод Subscribe поставщика и назначает возвращенный объект переменнойunsubscriber.public class TemperatureReporter : IObserver<Temperature> { private IDisposable unsubscriber; private bool first = true; private Temperature last; public virtual void Subscribe(IObservable<Temperature> provider) { unsubscriber = provider.Subscribe(this); }Public Class TemperatureReporter : Implements IObserver(Of Temperature) Private unsubscriber As IDisposable Private first As Boolean = True Private last As Temperature Public Overridable Sub Subscribe(ByVal provider As IObservable(Of Temperature)) unsubscriber = provider.Subscribe(Me) End SubОпределите метод, позволяющий наблюдателю перестать получать уведомления, прежде чем поставщик вызывает свою IObserver<T>.OnCompleted реализацию, если эта функция требуется. В следующем примере определяется метод
Unsubscribe.public virtual void Unsubscribe() { unsubscriber.Dispose(); }Public Overridable Sub Unsubscribe() unsubscriber.Dispose() End SubПредоставьте реализации трех методов, определенных интерфейсом IObserver<T>: IObserver<T>.OnNext, IObserver<T>.OnErrorи IObserver<T>.OnCompleted. В зависимости от поставщика и потребностей приложения, методы OnError и OnCompleted могут быть реализациями-заглушками. Обратите внимание, что метод OnError не должен обрабатывать переданный объект Exception как исключение, а метод OnCompleted может вызывать реализацию IDisposable.Dispose поставщика. В следующем примере показана реализация IObserver<T> класса
TemperatureReporter.public virtual void OnCompleted() { Console.WriteLine("Additional temperature data will not be transmitted."); } public virtual void OnError(Exception error) { // Do nothing. } public virtual void OnNext(Temperature value) { Console.WriteLine($"The temperature is {value.Degrees}°C at {value.Date:g}"); if (first) { last = value; first = false; } else { Console.WriteLine($" Change: {value.Degrees - last.Degrees}° in {value.Date.ToUniversalTime() - last.Date.ToUniversalTime():g}"); } }Public Overridable Sub OnCompleted() Implements System.IObserver(Of Temperature).OnCompleted Console.WriteLine("Additional temperature data will not be transmitted.") End Sub Public Overridable Sub OnError(ByVal [error] As System.Exception) Implements System.IObserver(Of Temperature).OnError ' Do nothing. End Sub Public Overridable Sub OnNext(ByVal value As Temperature) Implements System.IObserver(Of Temperature).OnNext Console.WriteLine("The temperature is {0}°C at {1:g}", value.Degrees, value.Date) If first Then last = value first = False Else Console.WriteLine(" Change: {0}° in {1:g}", value.Degrees - last.Degrees, value.Date.ToUniversalTime - last.Date.ToUniversalTime) End If End Sub
Пример
В следующем примере содержится полный исходный код для класса TemperatureReporter, который предоставляет реализацию IObserver<T> для приложения мониторинга температуры.
public class TemperatureReporter : IObserver<Temperature>
{
private IDisposable unsubscriber;
private bool first = true;
private Temperature last;
public virtual void Subscribe(IObservable<Temperature> provider)
{
unsubscriber = provider.Subscribe(this);
}
public virtual void Unsubscribe()
{
unsubscriber.Dispose();
}
public virtual void OnCompleted()
{
Console.WriteLine("Additional temperature data will not be transmitted.");
}
public virtual void OnError(Exception error)
{
// Do nothing.
}
public virtual void OnNext(Temperature value)
{
Console.WriteLine($"The temperature is {value.Degrees}°C at {value.Date:g}");
if (first)
{
last = value;
first = false;
}
else
{
Console.WriteLine($" Change: {value.Degrees - last.Degrees}° in {value.Date.ToUniversalTime() - last.Date.ToUniversalTime():g}");
}
}
}
Public Class TemperatureReporter : Implements IObserver(Of Temperature)
Private unsubscriber As IDisposable
Private first As Boolean = True
Private last As Temperature
Public Overridable Sub Subscribe(ByVal provider As IObservable(Of Temperature))
unsubscriber = provider.Subscribe(Me)
End Sub
Public Overridable Sub Unsubscribe()
unsubscriber.Dispose()
End Sub
Public Overridable Sub OnCompleted() Implements System.IObserver(Of Temperature).OnCompleted
Console.WriteLine("Additional temperature data will not be transmitted.")
End Sub
Public Overridable Sub OnError(ByVal [error] As System.Exception) Implements System.IObserver(Of Temperature).OnError
' Do nothing.
End Sub
Public Overridable Sub OnNext(ByVal value As Temperature) Implements System.IObserver(Of Temperature).OnNext
Console.WriteLine("The temperature is {0}°C at {1:g}", value.Degrees, value.Date)
If first Then
last = value
first = False
Else
Console.WriteLine(" Change: {0}° in {1:g}", value.Degrees - last.Degrees,
value.Date.ToUniversalTime - last.Date.ToUniversalTime)
End If
End Sub
End Class
См. также
- IObserver<T>
- шаблон проектирования «наблюдатель»
- Как реализовать провайдера
- Рекомендации по проектированию шаблонов наблюдателя