策略设计模式

来源:互联网 发布:导弹拦截 CSDN C语言 编辑:程序博客网 时间:2024/06/05 23:46

网上有许多关于策略设计模式的介绍,其实在策略设计模式就是一句话将执行策略封装成对象,不同的具体策略对象会对相同的输入产生不同的结果。前半句是策略设计模式,后半句就是多态。可以说策略设计模式离不开多态。首先简单地讲一下策略设计模式。
我觉得uml图是讲解设计模式最好的工具,直接上图。
策略设计模式uml图
策略设计模式如图无非就是通过一个上下文来调用具体策略的方法,代码如下:

    //策略接口      public Interface Strategy{                public void method();      }      //策略具体实现类1      public Class StrategyOne implements Strategy{               public void method(){.....}      }      //策略具体实现类2      public Class StrategyTwo implements Strategy{               public void method(){.....}      }      //Context类      public Class Context{          private Strategy strategy;          public Context(Strategy strategy){              this.strategy = strategy;          }                public void execute(){                    strategy.method();                }      }      //客户端调用      public static void main(String[] args){                  Strategy strategyOne = new StrategyOne();                  Context context = new Context(strategyOne);                  context.execute();      }

然而这种方式存在一个缺陷,在客户端实例化了策略类,导致代码的耦合度过高。我们是否可以考虑不在客户端直接实例化策略类呢。结合简单工厂方法,我们将Context类修改为如下:

  public Class Context {   public void execute(String type){             if(type.equals("策略一")){             new StrategyOne().method();             }else if(type.equals("策略二")){             new StrategyTwo().method();             }else{                     //.......             }   }   }

现在代码实现了策略类和客户端的解耦,然而事实上,
1:Context类和客户端并没有解耦
2:添加新的策略需要修改Context类(添加新的分支),不利于程序的扩展
首先解决第一个问题,我们开发常常使用Spring,这里就使用Spring的依赖注入,将Context交给Spring容器管理,这里就不介绍Spring了。这样其实连Context都和客户端解耦了。
接着解决第二个问题,我们可以考虑使用反射来解决这个问题,根据传进来的类型type决定实例化哪个策略对象。这的确是一种不错的方法,然而当使用Spring进行开发的时候,使用了反射就浪费了Spring本身的依赖注入功能了。所以我们就不选择反射了,那么不选择反射如何实现不修改分支就可以动态添加策略呢?最好的方式是在Context中new一个HashMap,将策略实例通过依赖注入的方式注入到这个map中。通过传入的参数决定key,通过key决定调用哪个策略对象。这样完美地将策略设计模式与Spring的依赖注入结合,也实现了完美的解耦。所以设计模式并不是死的,灵活的运用设计模式才是王道。