博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
设计模式之策略模式
阅读量:4093 次
发布时间:2019-05-25

本文共 1130 字,大约阅读时间需要 3 分钟。

一、什么是策略模式?

策略模式定义了算法族,分别封装起来,让它们之间可以互相替换。此模式让算法的变化独立于使用算法的客户。核心思想是将变化的部分抽离为独立维度。

二、从鸭子示例窥探策略模式的优势

  • 需求版本一:

           鸭子种类:绿头鸭、红头鸭、橡皮鸭

 行为:quack()---呱呱叫,swim()---游泳,display()---展示外观。

 附:橡皮鸭不会呱呱叫,而是吱吱叫。

 可能我们会这么设计:

  • 需求版本二:

          鸭子种类:绿头鸭,红头鸭,橡皮鸭,木头鸭

行为:quack(),swim(),display(),fly()---飞行

附:橡皮鸭会吱吱叫不会飞,木头鸭不会叫不会飞。其他的鸭子会呱呱叫会飞。

这时候我们有以下几种思路

1.将fly()方法放入抽象父类Duck中,这样所有的鸭子子类都需要实现fly()方法,即使它不会飞,虽然我们可以为不会飞的鸭子提供空的fly()方法实现。但是这种封装逻辑还是会很奇怪。

1.1将fly()实现放在Duck中的时候,对于可维护、可扩展性是不好的。(比如说我后面又需要区分会飞鸭子,进行不同 的飞行 处理,那么只能够在子类中覆盖fly()方法,这显然是不符合OCP原则的,长期如此系统会变得难以维护。其中RubberDuck提 供fly()的空实现,DecoyDuck提供quack()和fly()的空实现

1.2将fly()的抽象放到Duck中,实现放到子类中的时候,就会有大量重复的代码(红框),因为很多鸭子的飞行行为是完全一 致的。(其中RubberDuck提供fly()的空实现,DecoyDuck提供quack()和fly()的空实现)

2.将鸭子的不同行为抽象为接口

这样做也是有问题的。

其一是不易于扩展,以后我要是说swim()行为也有新品种的鸭子没法游泳。咋办,你还得动现在的类,不符合OCP,随着鸭子种类的增加,这种问题随时会出现。

其二是大量重复的代码,,接口相比于抽象类就是没有办法提供公有的实现。导致MallardDuck和RedheadDuck中有重复的代码,意味着你需要维护两份。

3.可能有些朋友第一反应是既然接口不行,我用抽象类啊,将Flyable和Quackable写成抽象类,将公有的实现放在里面不就结了?

然而Java不支持多继承。。就算是支持,如果以后要区分对待也是不好处理的。

4.将变化的和不变的分离,将不变的抽离出来,并用组合的方式建立与不变部分的关联,就是我们要讲的策略模式

这种做法看起来似乎很复杂。但事实上是极好的。后面只要有什么行为需要独立于鸭子使用,都可以扩展出一个新的族,现有的族也很好扩展。并且通过组合与Duck关联,耦合性低。

三、策略模式通用UML图

你可能感兴趣的文章
19 个 JavaScript 常用的简写技术
查看>>
iOS应用间相互跳转
查看>>
iOS开发之支付宝集成
查看>>
iOS开发 支付之银联支付集成
查看>>
iOS开发支付集成之微信支付
查看>>
浅谈JavaScript--声明提升
查看>>
React非嵌套组件通信
查看>>
Websocket 使用指南
查看>>
浏览器兼容性问题解决方案 · 总结
查看>>
一个很棒的Flutter学习资源列表
查看>>
为什么你应该放弃React老的Context API用新的Context API
查看>>
Koa2初体验
查看>>
Koa 2 初体验(二)
查看>>
Koa2框架原理解析和实现
查看>>
vue源码系列文章good
查看>>
你不知道的Virtual DOM
查看>>
VUE面试题总结
查看>>
写好JavaScript条件语句的5条守则
查看>>
原生JS中DOM节点相关API合集
查看>>
【TINY4412】U-BOOT移植笔记:(7)SDRAM驱动
查看>>