监视程序设计模式

观察者设计模式使订阅者能够向提供程序注册和接收通知。 它适用于需要基于推送的通知的任何方案。 该模式定义提供程序(也称为主体可观察对象)和零、一个或多个观察者。 观察程序向提供程序注册,每当发生预定义的条件、事件或状态更改时,提供程序都会通过调用委托自动通知所有观察程序。 在此方法调用中,提供程序还可以向观察者提供当前状态信息。 在 .NET 中,通过实现泛型 System.IObservable<T>System.IObserver<T> 接口来应用观察程序设计模式。 泛型类型参数表示提供通知信息的类型。

何时应用模式

观察者设计模式适用于基于推送的分布式通知,因为它支持在两个不同的组件或应用程序层(如数据源(业务逻辑)层和用户界面(显示)层之间实现干净分离。 每当提供程序使用回调向客户端提供当前信息时,都可以实现该模式。

实现模式需要提供以下详细信息:

  • 提供者或主题,即向观察者发送通知的对象。 提供者是实现接口的 IObservable<T> 类或结构体。 提供程序必须实现单个方法, IObservable<T>.Subscribe该方法由希望从提供程序接收通知的观察程序调用。

  • 观察者是一个接收提供程序通知的对象。 观察者是实现 IObserver<T> 接口的类或结构。 观察程序必须实现三种方法,所有这些方法均由提供程序调用:

  • 允许提供者跟踪观察者的机制。 通常,提供程序使用容器对象(如 System.Collections.Generic.List<T> 对象)来保存对 IObserver<T> 已订阅通知的实现的引用。 使用存储容器实现此目的,使提供者可以处理零到无限数量的观察者。 未定义观察程序接收通知的顺序;提供程序可以使用任何方法来确定顺序。

  • 一种 IDisposable 实现,允许提供程序在通知完成后删除观察者。 观察程序从IDisposable该方法接收对Subscribe实现的引用,因此,在提供程序完成发送通知之前,观察程序还可以调用IDisposable.Dispose该方法取消订阅。

  • 一个对象,包含提供者发送到其观察者的数据。 此对象的类型对应于IObservable<T>IObserver<T>接口的泛型类型参数。 尽管此对象可以与 IObservable<T> 实现相同,但最常见的是单独的类型。

注释

除了实现观察程序设计模式之外,你可能还有兴趣探索使用 IObservable<T>IObserver<T> 接口生成的库。 例如, 用于 .NET 的被动扩展(Rx) 由一组扩展方法和 LINQ 标准序列运算符组成,以支持异步编程。

应用该模式

以下示例使用观察者设计模式来实现机场行李索赔信息系统。 BaggageInfo 类提供有关到达航班以及可领取每次航班行李的行李传送带的信息。 它显示在以下示例中。

namespace Observables.Example;

public readonly record struct BaggageInfo(
    int FlightNumber,
    string From,
    int Carousel);
Public Class BaggageInfo
    Private flightNo As Integer
    Private origin As String
    Private ___location As Integer

    Friend Sub New(ByVal flight As Integer, ByVal from As String, ByVal carousel As Integer)
        Me.flightNo = flight
        Me.origin = from
        Me.___location = carousel
    End Sub

    Public ReadOnly Property FlightNumber As Integer
        Get
            Return Me.flightNo
        End Get
    End Property

    Public ReadOnly Property From As String
        Get
            Return Me.origin
        End Get
    End Property

    Public ReadOnly Property Carousel As Integer
        Get
            Return Me.___location
        End Get
    End Property
End Class

BaggageHandler 类负责接收有关到达航班和行李认领传送带的信息。 在内部,它维护两个集合:

  • _observers:观察更新信息的客户端集合。
  • _flights:航班及其指定行李传送带的集合。

以下示例显示了该类的 BaggageHandler 源代码。

namespace Observables.Example;

public sealed class BaggageHandler : IObservable<BaggageInfo>
{
    private readonly HashSet<IObserver<BaggageInfo>> _observers = new();
    private readonly HashSet<BaggageInfo> _flights = new();

    public IDisposable Subscribe(IObserver<BaggageInfo> observer)
    {
        // Check whether observer is already registered. If not, add it.
        if (_observers.Add(observer))
        {
            // Provide observer with existing data.
            foreach (BaggageInfo item in _flights)
            {
                observer.OnNext(item);
            }
        }

        return new Unsubscriber<BaggageInfo>(_observers, observer);
    }

    // Called to indicate all baggage is now unloaded.
    public void BaggageStatus(int flightNumber) =>
        BaggageStatus(flightNumber, string.Empty, 0);

    public void BaggageStatus(int flightNumber, string from, int carousel)
    {
        var info = new BaggageInfo(flightNumber, from, carousel);

        // Carousel is assigned, so add new info object to list.
        if (carousel > 0 && _flights.Add(info))
        {
            foreach (IObserver<BaggageInfo> observer in _observers)
            {
                observer.OnNext(info);
            }
        }
        else if (carousel is 0)
        {
            // Baggage claim for flight is done.
            if (_flights.RemoveWhere(
                flight => flight.FlightNumber == info.FlightNumber) > 0)
            {
                foreach (IObserver<BaggageInfo> observer in _observers)
                {
                    observer.OnNext(info);
                }
            }
        }
    }

    public void LastBaggageClaimed()
    {
        foreach (IObserver<BaggageInfo> observer in _observers)
        {
            observer.OnCompleted();
        }

        _observers.Clear();
    }
}
Public Class BaggageHandler : Implements IObservable(Of BaggageInfo)

    Private observers As List(Of IObserver(Of BaggageInfo))
    Private flights As List(Of BaggageInfo)

    Public Sub New()
        observers = New List(Of IObserver(Of BaggageInfo))
        flights = New List(Of BaggageInfo)
    End Sub

    Public Function Subscribe(ByVal observer As IObserver(Of BaggageInfo)) As IDisposable _
                    Implements IObservable(Of BaggageInfo).Subscribe
        ' Check whether observer is already registered. If not, add it
        If Not observers.Contains(observer) Then
            observers.Add(observer)
            ' Provide observer with existing data.
            For Each item In flights
                observer.OnNext(item)
            Next
        End If
        Return New Unsubscriber(Of BaggageInfo)(observers, observer)
    End Function

    ' Called to indicate all baggage is now unloaded.
    Public Sub BaggageStatus(ByVal flightNo As Integer)
        BaggageStatus(flightNo, String.Empty, 0)
    End Sub

    Public Sub BaggageStatus(ByVal flightNo As Integer, ByVal from As String, ByVal carousel As Integer)
        Dim info As New BaggageInfo(flightNo, from, carousel)

        ' Carousel is assigned, so add new info object to list.
        If carousel > 0 And Not flights.Contains(info) Then
            flights.Add(info)
            For Each observer In observers
                observer.OnNext(info)
            Next
        ElseIf carousel = 0 Then
            ' Baggage claim for flight is done
            Dim flightsToRemove As New List(Of BaggageInfo)
            For Each flight In flights
                If info.FlightNumber = flight.FlightNumber Then
                    flightsToRemove.Add(flight)
                    For Each observer In observers
                        observer.OnNext(info)
                    Next
                End If
            Next
            For Each flightToRemove In flightsToRemove
                flights.Remove(flightToRemove)
            Next
            flightsToRemove.Clear()
        End If
    End Sub

    Public Sub LastBaggageClaimed()
        For Each observer In observers
            observer.OnCompleted()
        Next
        observers.Clear()
    End Sub
End Class

希望接收更新信息的客户端调用 BaggageHandler.Subscribe 该方法。 如果客户端以前未订阅通知,则将对客户端的 IObserver<T> 实现的引用添加到 _observers 集合中。

可以调用重载 BaggageHandler.BaggageStatus 方法,以指示航班中的行李正在卸载或不再卸载。 在第一种情况下,该方法传递航班号、航班起飞机场和正在卸载行李的传送带。 第二种情况下,该方法仅传递一个航班号。 对于正在卸载的行李,该方法检查传递到方法的 BaggageInfo 信息是否存在于 _flights 集合中。 如果不存在,该方法将添加信息,并调用每个观察者的 OnNext 方法。 对于行李不再卸载的航班,该方法会检查该航班的信息是否存储在_flights集合中。 如果是这样,该方法将调用每个观察者的OnNext 方法,并从BaggageInfo 集合中删除_flights 对象。

当当天的最后一班航班降落并处理行李时,将调用该方法 BaggageHandler.LastBaggageClaimed 。 此方法调用每个观察者 OnCompleted 的方法以指示所有通知都已完成,然后清除 _observers 集合。

提供程序 Subscribe 的方法返回一个 IDisposable 实现,使观察者能够在调用 OnCompleted 方法之前停止接收通知。 此 Unsubscriber(Of BaggageInfo) 类的源代码在以下示例中显示。 此类在 BaggageHandler.Subscribe 方法中实例化时,它将传递对 _observers 集合的引用以及对添加到集合中的观察者的引用。 这些引用将分配给局部变量。 调用对象的 Dispose 方法时,它会检查观察程序是否仍存在于集合中 _observers ,如果仍然存在,则删除观察程序。

namespace Observables.Example;

internal sealed class Unsubscriber<BaggageInfo> : IDisposable
{
    private readonly ISet<IObserver<BaggageInfo>> _observers;
    private readonly IObserver<BaggageInfo> _observer;

    internal Unsubscriber(
        ISet<IObserver<BaggageInfo>> observers,
        IObserver<BaggageInfo> observer) => (_observers, _observer) = (observers, observer);

    public void Dispose() => _observers.Remove(_observer);
}
Friend Class Unsubscriber(Of BaggageInfo) : Implements IDisposable
    Private _observers As List(Of IObserver(Of BaggageInfo))
    Private _observer As IObserver(Of BaggageInfo)

    Friend Sub New(ByVal observers As List(Of IObserver(Of BaggageInfo)), ByVal observer As IObserver(Of BaggageInfo))
        Me._observers = observers
        Me._observer = observer
    End Sub

    Public Sub Dispose() Implements IDisposable.Dispose
        If _observers.Contains(_observer) Then
            _observers.Remove(_observer)
        End If
    End Sub
End Class

以下示例提供一个名为IObserver<T>ArrivalsMonitor的实现,该实现是显示行李索赔信息的基类。 信息按字母顺序显示,按原始城市的名称显示。 将 ArrivalsMonitor 的方法标记为 overridable(在 Visual Basic 中)或 virtual(在 C# 中),因此它们可在派生类中重写。

namespace Observables.Example;

public class ArrivalsMonitor : IObserver<BaggageInfo>
{
    private readonly string _name;
    private readonly List<string> _flights = new();
    private readonly string _format = "{0,-20} {1,5}  {2, 3}";
    private IDisposable? _cancellation;    

    public ArrivalsMonitor(string name)
    {
        ArgumentException.ThrowIfNullOrEmpty(name);
        _name = name;
    }

    public virtual void Subscribe(BaggageHandler provider) =>
        _cancellation = provider.Subscribe(this);

    public virtual void Unsubscribe()
    {
        _cancellation?.Dispose();
        _flights.Clear();
    }

    public virtual void OnCompleted() => _flights.Clear();

    // No implementation needed: Method is not called by the BaggageHandler class.
    public virtual void OnError(Exception e)
    {
        // No implementation.
    }

    // Update information.
    public virtual void OnNext(BaggageInfo info)
    {
        bool updated = false;

        // Flight has unloaded its baggage; remove from the monitor.
        if (info.Carousel is 0)
        {
            string flightNumber = string.Format("{0,5}", info.FlightNumber);
            for (int index = _flights.Count - 1; index >= 0; index--)
            {
                string flightInfo = _flights[index];
                if (flightInfo.Substring(21, 5).Equals(flightNumber))
                {
                    updated = true;
                    _flights.RemoveAt(index);
                }
            }
        }
        else
        {
            // Add flight if it doesn't exist in the collection.
            string flightInfo = string.Format(_format, info.From, info.FlightNumber, info.Carousel);
            if (_flights.Contains(flightInfo) is false)
            {
                _flights.Add(flightInfo);
                updated = true;
            }
        }

        if (updated)
        {
            _flights.Sort();
            Console.WriteLine($"Arrivals information from {_name}");
            foreach (string flightInfo in _flights)
            {
                Console.WriteLine(flightInfo);
            }

            Console.WriteLine();
        }
    }
}
Public Class ArrivalsMonitor : Implements IObserver(Of BaggageInfo)
    Private name As String
    Private flightInfos As New List(Of String)
    Private cancellation As IDisposable
    Private fmt As String = "{0,-20} {1,5}  {2, 3}"

    Public Sub New(ByVal name As String)
        If String.IsNullOrEmpty(name) Then Throw New ArgumentNullException("The observer must be assigned a name.")

        Me.name = name
    End Sub

    Public Overridable Sub Subscribe(ByVal provider As BaggageHandler)
        cancellation = provider.Subscribe(Me)
    End Sub

    Public Overridable Sub Unsubscribe()
        cancellation.Dispose()
        flightInfos.Clear()
    End Sub

    Public Overridable Sub OnCompleted() Implements System.IObserver(Of BaggageInfo).OnCompleted
        flightInfos.Clear()
    End Sub

    ' No implementation needed: Method is not called by the BaggageHandler class.
    Public Overridable Sub OnError(ByVal e As System.Exception) Implements System.IObserver(Of BaggageInfo).OnError
        ' No implementation.
    End Sub

    ' Update information.
    Public Overridable Sub OnNext(ByVal info As BaggageInfo) Implements System.IObserver(Of BaggageInfo).OnNext
        Dim updated As Boolean = False

        ' Flight has unloaded its baggage; remove from the monitor.
        If info.Carousel = 0 Then
            Dim flightsToRemove As New List(Of String)
            Dim flightNo As String = String.Format("{0,5}", info.FlightNumber)
            For Each flightInfo In flightInfos
                If flightInfo.Substring(21, 5).Equals(flightNo) Then
                    flightsToRemove.Add(flightInfo)
                    updated = True
                End If
            Next
            For Each flightToRemove In flightsToRemove
                flightInfos.Remove(flightToRemove)
            Next
            flightsToRemove.Clear()
        Else
            ' Add flight if it does not exist in the collection.
            Dim flightInfo As String = String.Format(fmt, info.From, info.FlightNumber, info.Carousel)
            If Not flightInfos.Contains(flightInfo) Then
                flightInfos.Add(flightInfo)
                updated = True
            End If
        End If
        If updated Then
            flightInfos.Sort()
            Console.WriteLine("Arrivals information from {0}", Me.name)
            For Each flightInfo In flightInfos
                Console.WriteLine(flightInfo)
            Next
            Console.WriteLine()
        End If
    End Sub
End Class

ArrivalsMonitor 类包括 SubscribeUnsubscribe 方法。 Subscribe 方法使类能够将通过调用 IDisposable 返回的 Subscribe 实现保存到私有变量中。 该方法 Unsubscribe 允许类通过调用提供程序的 Dispose 实现取消订阅通知。 ArrivalsMonitor还提供了OnNextOnErrorOnCompleted方法的实现。 只有OnNext 实现包含大量代码。 该方法与一个专用的、已排序的通用 List<T> 对象配合使用,该对象维护有关抵达航班的始发机场的信息以及其行李可用的传送带。 BaggageHandler如果类报告新的航班到达,则OnNext方法实现会将有关航班的信息添加到列表中。 如果 BaggageHandler 类报告已卸载该航班的行李,则 OnNext 方法从列表中移除该航班。 每当进行更改时,列表将排序并显示到控制台。

以下示例包含应用程序入口点,在此实例化 BaggageHandler 类并创建两个 ArrivalsMonitor 类的实例,使用 BaggageHandler.BaggageStatus 方法添加和删除关于到达航班的信息。 在每个情况下,观察者都会收到更新并正确显示行李索赔信息。

using Observables.Example;

BaggageHandler provider = new();
ArrivalsMonitor observer1 = new("BaggageClaimMonitor1");
ArrivalsMonitor observer2 = new("SecurityExit");

provider.BaggageStatus(712, "Detroit", 3);
observer1.Subscribe(provider);

provider.BaggageStatus(712, "Kalamazoo", 3);
provider.BaggageStatus(400, "New York-Kennedy", 1);
provider.BaggageStatus(712, "Detroit", 3);
observer2.Subscribe(provider);

provider.BaggageStatus(511, "San Francisco", 2);
provider.BaggageStatus(712);
observer2.Unsubscribe();

provider.BaggageStatus(400);
provider.LastBaggageClaimed();

// Sample output:
//   Arrivals information from BaggageClaimMonitor1
//   Detroit                712    3
//   
//   Arrivals information from BaggageClaimMonitor1
//   Detroit                712    3
//   Kalamazoo              712    3
//   
//   Arrivals information from BaggageClaimMonitor1
//   Detroit                712    3
//   Kalamazoo              712    3
//   New York-Kennedy       400    1
//   
//   Arrivals information from SecurityExit
//   Detroit                712    3
//   
//   Arrivals information from SecurityExit
//   Detroit                712    3
//   Kalamazoo              712    3
//   
//   Arrivals information from SecurityExit
//   Detroit                712    3
//   Kalamazoo              712    3
//   New York-Kennedy       400    1
//   
//   Arrivals information from BaggageClaimMonitor1
//   Detroit                712    3
//   Kalamazoo              712    3
//   New York-Kennedy       400    1
//   San Francisco          511    2
//   
//   Arrivals information from SecurityExit
//   Detroit                712    3
//   Kalamazoo              712    3
//   New York-Kennedy       400    1
//   San Francisco          511    2
//   
//   Arrivals information from BaggageClaimMonitor1
//   New York-Kennedy       400    1
//   San Francisco          511    2
//   
//   Arrivals information from SecurityExit
//   New York-Kennedy       400    1
//   San Francisco          511    2
//   
//   Arrivals information from BaggageClaimMonitor1
//   San Francisco          511    2
Module Example
    Public Sub Main()
        Dim provider As New BaggageHandler()
        Dim observer1 As New ArrivalsMonitor("BaggageClaimMonitor1")
        Dim observer2 As New ArrivalsMonitor("SecurityExit")

        provider.BaggageStatus(712, "Detroit", 3)
        observer1.Subscribe(provider)
        provider.BaggageStatus(712, "Kalamazoo", 3)
        provider.BaggageStatus(400, "New York-Kennedy", 1)
        provider.BaggageStatus(712, "Detroit", 3)
        observer2.Subscribe(provider)
        provider.BaggageStatus(511, "San Francisco", 2)
        provider.BaggageStatus(712)
        observer2.Unsubscribe()
        provider.BaggageStatus(400)
        provider.LastBaggageClaimed()
    End Sub
End Module
' The example displays the following output:
'      Arrivals information from BaggageClaimMonitor1
'      Detroit                712    3
'
'      Arrivals information from BaggageClaimMonitor1
'      Detroit                712    3
'      Kalamazoo              712    3
'
'      Arrivals information from BaggageClaimMonitor1
'      Detroit                712    3
'      Kalamazoo              712    3
'      New York-Kennedy       400    1
'
'      Arrivals information from SecurityExit
'      Detroit                712    3
'
'      Arrivals information from SecurityExit
'      Detroit                712    3
'      Kalamazoo              712    3
'
'      Arrivals information from SecurityExit
'      Detroit                712    3
'      Kalamazoo              712    3
'      New York-Kennedy       400    1
'
'      Arrivals information from BaggageClaimMonitor1
'      Detroit                712    3
'      Kalamazoo              712    3
'      New York-Kennedy       400    1
'      San Francisco          511    2
'
'      Arrivals information from SecurityExit
'      Detroit                712    3
'      Kalamazoo              712    3
'      New York-Kennedy       400    1
'      San Francisco          511    2
'
'      Arrivals information from BaggageClaimMonitor1
'      New York-Kennedy       400    1
'      San Francisco          511    2
'
'      Arrivals information from SecurityExit
'      New York-Kennedy       400    1
'      San Francisco          511    2
'
'      Arrivals information from BaggageClaimMonitor1
'      San Francisco          511    2
标题 DESCRIPTION
观察者设计模式最佳做法 介绍在开发实现观察程序设计模式的应用程序时要采用的最佳做法。
如何:实现提供程序 为温度监控应用程序提供一个提供程序的分步实现。
如何:实现监视程序 为温度监控应用程序提供一个观察者的分步实现。