관찰 모드라고 하면 정원에서 많이 볼 수 있을 것 같아요. 따라서 이 블로그를 작성하는 목적은 두 가지입니다.
1. 관찰자 패턴은 느슨하게 결합된 코드를 작성하는 데 필요한 패턴입니다. 그 중요성은 코드 수준에 관계없이 많은 구성 요소에서 Publish -Subscribe를 사용한다는 점에서 자명합니다. 모드이므로 본인의 이해에 따라 사용 시나리오를 다시 설계하고 그 안에 관찰자 모드를 유연하게 사용하고 싶습니다
2. 관찰자 모드를 C#으로 구현하기 위한 세 가지 솔루션을 요약하고 싶지만 현재는 아니요 이 요약을 봅니다
이제 이러한 시나리오를 가정하고 관찰 모드를 사용하여 요구 사항을 실현해 보겠습니다.
미래에는 스마트 홈이 모든 가정에 도입되고 모든 가정에 API가 있게 됩니다. 고객이 통합을 맞춤화할 수 있도록 최초의 스마트 알람시계(smartClock)가 출시되었습니다. 제조업체는 알람 시간이 설정되면 이때 알림을 보냅니다. 스마트 우유 히터, 빵 굽는 기계, 치약 짜는 장비 모두 이 알람 시계 메시지를 구독해야 주인을 위해 우유, 빵, 치약 등을 자동으로 준비합니다.
이 시나리오는 아주 전형적인 관찰자 모드입니다. 스마트 알람시계의 알람시계는 관찰자만 있으면 됩니다. 테마는 느슨하게 결합된 코딩 모델을 가능하게 합니다. 이 요구사항을 세 가지 옵션을 통해 하나씩 구현해 보겠습니다.
1. .net의 Event 모델을 사용하여 구현
.net의 Event 모델은 .net 탄생 이후 코드에서 널리 사용되는 일반적인 관찰자 패턴입니다. 이 시나리오에서는 이벤트 모델이 사용됩니다.
제조사에서는 매우 간단한 API 세트를 제공합니다
public void SetAlarmTime(TimeSpan timeSpan) { _alarmTime = _now().Add(timeSpan); RunBackgourndRunner(_now, _alarmTime); }
protected void RunBackgourndRunner(Func<DateTime> now,DateTime? alarmTime ) { if (alarmTime.HasValue) { var cancelToken = new CancellationTokenSource(); var task = new Task(() => { while (!cancelToken.IsCancellationRequested) { if (now.AreEquals(alarmTime.Value)) { //闹铃时间到了 ItIsTimeToAlarm(); cancelToken.Cancel(); } cancelToken.Token.WaitHandle.WaitOne(TimeSpan.FromSeconds(2)); } }, cancelToken.Token, TaskCreationOptions.LongRunning); task.Start(); } }
var args = new AlarmEventArgs(_alarmTime.Value, 0.92m); OnAlarmEvent(args);
public virtual void OnAlarm(AlarmEventArgs e) { if(Alarm!=null) Alarm(this,e); }
스마트 알람시계(SmartClock)가 구현되었습니다. 우유 히터(MilkSchedule)에서 다음 알람 메시지를 구독합니다.
public void PrepareMilkInTheMorning() { _clock.Alarm += (clock, args) => { Message = "Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith( args.AlarmTime, args.ElectricQuantity*100); Console.WriteLine(Message); }; _clock.SetAlarmTime(TimeSpan.FromSeconds(2)); }
public IDisposable Subscribe(IObserver<AlarmData> observer) { if (!_observers.Contains(observer)) { _observers.Add(observer); } return new DisposedAction(() => _observers.Remove(observer)); }
public override void ItIsTimeToAlarm() { var alarm = new AlarmData(_alarmTime.Value, 0.92m); _observers.ForEach(o=>o.OnNext(alarm)); }
public void Subscribe(TimeSpan timeSpan) { _unSubscriber = _clock.Subscribe(this); _clock.SetAlarmTime(timeSpan); } public void Unsubscribe() { _unSubscriber.Dispose(); } public void OnNext(AlarmData value) { Message = "Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith( value.AlarmTime, value.ElectricQuantity * 100); Console.WriteLine(Message); }
var milkSchedule = new MilkSchedule(); //Act milkSchedule.Subscribe(TimeSpan.FromSeconds(12));
这种方案中,智能闹钟(smartClock)提供的API需要设计成这样:
public void SetAlarmTime(TimeSpan timeSpan,Action<AlarmData> alarmAction) { _alarmTime = _now().Add(timeSpan); _alarmAction = alarmAction; RunBackgourndRunner(_now, _alarmTime); }
方法签名中要接受一个Action
public override void ItIsTimeToAlarm() { if (_alarmAction != null) { var alarmData = new AlarmData(_alarmTime.Value, 0.92m); _alarmAction(alarmData); } }
牛奶加热器中使用这种API也很简单:
_clock.SetAlarmTime(TimeSpan.FromSeconds(1), (data) => { Message = "Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith( data.AlarmTime, data.ElectricQuantity * 100); });
在实际使用过程中我会把这种API设计成fluent模型,调用起来代码更清晰:
智能闹钟(smartClock)中的API:
public Clock SetAlarmTime(TimeSpan timeSpan) { _alarmTime = _now().Add(timeSpan); RunBackgourndRunner(_now, _alarmTime); return this; } public void OnAlarm(Action<AlarmData> alarmAction) { _alarmAction = alarmAction; }
牛奶加热器中进行调用:
_clock.SetAlarmTime(TimeSpan.FromSeconds(2)) .OnAlarm((data) => { Message = "Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith( data.AlarmTime, data.ElectricQuantity * 100); });
显然改进后的写法语义更好:闹钟.设置闹铃时间().当报警时(()=>{执行以下功能})
这种函数式写法更简练,但是也有明显的缺点,该模型不支持多个观察者,当面包烘烤机使用这样的API时,会覆盖牛奶加热器的函数,即每次只支持一个观察者使用。
结束语,本文总结了.net下的三种观察者模型实现方案,能在编程场景下选择最合适的模型当然是我们的最终目标。