zoukankan      html  css  js  c++  java
  • Head First 设计模式读书笔记(1)-策略模式

    一、策略模式的定义

    策略模式定义了算法族,分别封装起来,让它们之间可以互换替换,此模式让算法的变化独立使用算法的客户。

    二、使用策略模式的一个例子

    2.1引出问题

    某公司做了一套模拟鸭子的游戏:该游戏会出现各种鸭子,他们具有相似点(都会叫,会游泳,会表达自己的特征)和异同点(每种展示自己的特征是不同的方法),很容易想到使用OO技术,定义个鸭子超类,让其他的鸭子类来继承鸭子超类。

    由于异同点都会展示自己的特征,只是展示特征方式不同,所以在超类中特征的方法是抽象或者使用一个接口(在此就使用接口,因为每一个类都会有展示方法)。下面给出其类图。

    image

    可当某一天发现了问题,

    问题1:如果出现了一个DuckC,展示方式和DuckA相同,都是使用相同的方法,那么就会在DuckC类中写了同样的代码,如果那一天想修改他们中的使用相同的方法,就要一个个去修改,如果是有多个的话,就容易出现问题,而且不能到达方法复用的效果。

    问题2:如果想让一只鸭子需要有两种不同的展示方式,可能随时改变展示方式,上面的设计就需要修改了

    2.2分析问题

    问题1为了达到复用,是不可能把所有的展示方式放到Duck类中,因为对应DuckA和DuckB以及DuckC可以随便调用方法,显然是不明智的。

    先把问题1放一放,我们来看一下问题2的,可以随时改变,C#中有个属性的思想不就是可以随时改变变量的值。是不是可以使用相同的方法来实现呢。每次使用set时,是让值赋给指定的变量,而且可以达到复用的目的(如Person p=new Person();p.Age=22,22是一个int类型中的一个对象,可以多次使用,而且还可以赋值为23)。一句简单的属性赋值,隐含着多种设计原则。

    原则一:找出应用中可能需要变化之处,把它们独立出来,不要和那些不需要变化的代码混在一起。

    在上面一段简单的代码中,我们可以把22,23看成计算机为我们封装好的类型。22和23是被独立出来的会变化的代码。具体的实现我们不需要知道怎么实现。只需要知道表示的一个数。

    原则二:针对接口编程,而不是针对实现编程。

    先简单的说一下上面的意思:让变的抽象成接口或者抽象类,然后通过单独的类来把接口实现或继承抽象类来实现,而不是将功能直接在代码中实现。

    在上面具体的23和22怎么的实现我们是没有直接写出new int();而是直接获取的类型,其实现交给计算机去处理。

    原则三:多使用组合,少使用继承

    为什么能随时“改变age”,这就是组合的力量,如果是使用继承的话,就不能在一个过程中改变age。

    上面使用整型可能有些不妥,仅仅作为一中理解形式。如果非要找一下策略模式在.net中的踪影,可以看一下ArrayList的Sort(IComparer comparer)方法,可以参考http://www.cnblogs.com/justinw/archive/2007/02/06/641414.html

    有了这三个原则,我觉得上面的两个问题应该可以解决了。

    2.3解决问题

    使用组合和属性的设计方法来处理

    image

    下面列出测试代码:

             Duck dA = new DuckA(); 
              IDisplay displayA= dA.GetDisplay(); 
              displayA.Display(); 
              Console.WriteLine("A类鸭子展示方法:");
    
               Console.WriteLine("================================");
    
              Duck dB = new DuckB(); 
              IDisplay displayB = dB.GetDisplay(); 
              Console.WriteLine("B类鸭子展示方法:");
    
              displayB.Display(); 
               Console.WriteLine("================================");
    
              Duck dC = new DuckC(); 
              IDisplay displayC = dC.GetDisplay();
    
              Console.WriteLine("C类鸭子展示方法:");
    
              displayC.Display(); 
              Console.WriteLine("================================"); 
              dC.SetDisplay(new DisplayClassA()); 
              Console.WriteLine("C类鸭子改变方法后展示方法:"); 
              dC.GetDisplay().Display(); 
              Console.Read(); 
    View Code

    输出结果:image

    三、策略模式对应的UML

    image

    该模式也体现了面向对象的基本特征:封装(把算法一个个封装起来),继承(接口的实现也可以看成是实现类“继承”了接口),多态(在Context调用接口的时间,开始并不知道接口的方法的具体实现,但是知道有个方法,使用的时间就可获取,调用)。

    源码  

    四、小结

    本文主要通过鸭子的设计,引出问题,分析问题,解决问题来体现了策略模式的灵活性,在分析过程中提出了三个设计原则,结合属性的实现思想来解决了问题。本文是属于读书笔记,可能在理解的地方有偏差,希望能够多多指教!

  • 相关阅读:
    区块链开发之Corda训练营笔记12:Flow习题答案
    区块链开发之Corda训练营笔记11:Flow代码演示
    gitolite的部署
    python操作redis
    python多进程
    mysql索引
    mysql事务
    mysql外键
    mysql联合查询
    python常用模块之re模块(正则)
  • 原文地址:https://www.cnblogs.com/lzhp/p/3344776.html
Copyright © 2011-2022 走看看