一个插排引发的设计思想 (一) 观察者模式
...待续....
不知道聊到设计模式, 经常给人两种感觉:
1. 原来这个就是A设计模式呀, 我之前也经常这么干, 就是到现在才知道A设计模式指的就是这个.
2. 这个B设计模式, 虽然书上讲的晦涩难懂, 但我还是勉强理解了, 可是平时基本上用不着, 不知道啥时候改用, 甚至有时候感觉这是在"过度设计".
今天就以一个插排为例来展开学习一下.
需求: 实现一个插排( 就叫它Output吧 , 英文Socket容易引起歧义 ) 的功能 , 它有N组插孔, 支持多个电器的插头(对应着叫Input吧)插入并供电.
如下图
............
需求分析:
1. 插排中有一个集合(即多组插孔), 并提供插头插入(Add)和拔出(Remove)的方法.
2. 插入和拔出的插头(参数)需要满足一定的规则, 大家都知道不同国家的插座标准好多不一样的,比如电压和针脚的间距等, 这里就按中国标准吧, 咱也弄个GB.
好的, 根据第一感觉开始coding....
一. 首先定义一个国标(GB)标准, 只有符合此标准的电器才能插入. abstract的"电器"类GBElectricalAppliance.
它有个插头的功能即 Input 方法,Input方法有两个参数 left 和 right 象征插头的两个针脚.
1 public abstract class GBElectricalAppliance 2 { 3 public abstract void Input(int left, int right); 4 }
二. 写个插排(Output)类
1 public class OutPut 2 { 3 public OutPut() 4 { 5 this.EACollection = new List<GBElectricalAppliance>(); 6 } 7 private List<GBElectricalAppliance> EACollection; 8 public void powered(int left,int right) 9 { 10 foreach (var item in EACollection) 11 { 12 item.Input(left,right); 13 } 14 } 15 public void AddInput(GBElectricalAppliance item) 16 { 17 EACollection.Add(item); 18 } 19 20 public void RemoveInput(GBElectricalAppliance item) 21 { 22 EACollection.Remove(item); 23 } 24 }
此类有个private 的集合 EACollection, 两个Public的方法AddInput和RemoveInput 方法参数为GBElectricalAppliance, 即只支持符合标准的插头插拔.
当插排Output供电(方法powered)时, 遍历集和EACollection, 调用集和中的电器的Input方法为电器供电.
三. 分别定义一个电视TV和电水壶ElectricKettle继承此类并实现插头的功能即 Input 方法.
1 public class TV : GBElectricalAppliance 2 { 3 public override void Input(int left, int right) 4 { 5 Show(); 6 Sound(); 7 } 8 9 private void Show() 10 { 11 Console.WriteLine("I am showing"); 12 } 13 private void Sound() 14 { 15 Console.WriteLine("I am sounding"); 16 } 17 } 18 19 public class ElectricKettle : GBElectricalAppliance 20 { 21 public override void Input(int left, int right) 22 { 23 Heat(); 24 } 25 26 private void Heat() 27 { 28 Console.WriteLine("I am heating"); 29 } 30 }
Main方法测试一下:
输出:
I am showing
I am sounding
I am heating
四. 小结
到此为止, 功能基本实现了, 我们还可以定义充电宝、电灯等"电器"继承GBElectricalAppliance插入.
仔细回味一下, 这其实是用到了观察者模式,
观察者(电器)将插头注册到被观察者(插排)的集和中, 当被插排供电后, 插排会通知所有注册到集和中的插头进行供电.
插排不关心插上的具体是什么电器, 只要它是符合国标的电器即可. 通过这样的设计尽量降低插座与电器之间的耦合度.
以上写法差不多是看到需求后按照第一感觉实现的, 仔细想想我们是不是可以用接口、委托什么的改造一下,
下一篇文章我们继续讨论.