1 编写鸭子项目,具体要求如下:

1) 有各种鸭子(比如 野鸭、北京鸭、水鸭等, 鸭子有各种行为,比如 叫、飞行等)

2) 显示鸭子的信息

2 传统方案解决鸭子问题的分析和代码实现

1) 传统的设计方案(类图)

2)代码实现

  1. package com.lin.strategy;
  2.  
  3. public abstract class Duck {
  4.  
  5. public abstract void display();
  6.  
  7. public void quack() {
  8. System.out.println("鸭子嘎嘎嘎嘎");
  9. }
  10.  
  11. public void swimming() {
  12. System.out.println("鸭子会游泳");
  13. }
  14.  
  15. public void fly() {
  16. System.out.println("鸭子会飞");
  17. }
  18. }
  1. package com.lin.strategy;
  2.  
  3. public class PekingDuck extends Duck {
  4.  
  5. @Override
  6. public void display() {
  7. System.out.println("这是北京鸭");
  8.  
  9. }
  10.  
  11. // 北京鸭不好飞翔
  12. @Override
  13. public void fly() {
  14. System.out.println("北京鸭不会飞翔");
  15. }
  16.  
  17. }
  1. package com.lin.strategy;
  2.  
  3. public class ToyDuck extends Duck {
  4.  
  5. @Override
  6. public void display() {
  7. System.out.println("玩具鸭");
  8.  
  9. }
  10.  
  11. // 要重写所有父类的方法
  12. public void quack() {
  13. System.out.println("鸭子不会嘎嘎嘎嘎");
  14. }
  15.  
  16. public void swimming() {
  17. System.out.println("鸭子不会游泳");
  18. }
  19.  
  20. public void fly() {
  21. System.out.println("鸭子不会飞");
  22. }
  23.  
  24. }
  1. package com.lin.strategy;
  2.  
  3. public class WildDuck extends Duck{
  4.  
  5. @Override
  6. public void display() {
  7. System.out.println("这是野鸭!");
  8. }
  9.  
  10. }

3 传统的方式实现的问题分析和解决方案

1) 其它鸭子,都继承了 Duck 类,所以 fly 让所有子类都会飞了,这是不正确的

2) 上面说的 1 的问题,其实是继承带来的问题:对类的局部改动,尤其超类的局部改动,会影响其他部分。会有溢出效应

3) 为了改进 1 问题,我们可以通过覆盖 fly  方法来解决 => 覆盖解决

4) 问题又来了,如果我们有一个玩具鸭子 ToyDuck, 这样就需要 ToyDuck 去覆盖 Duck 的所有实现的方法 => 解决思路 -》 策略模式 (strategy pattern)

4 策略模式基本介绍

1) 策略模式(Strategy Pattern)中,定义算法族(策略组),分别封装起来,让他们之间可以互相替换,此模式让算法的变化独立于使用算法的客户

2) 这算法体现了几个设计原则,第一、把变化的代码从不变的代码中分离出来;第二、针对接口编程而不是具体类(定义了策略接口);第三、多用组合/聚合,少用继承(客户通过组合方式使用策略)。

5 策略模式的原理类图

说明:从上图可以看到,客户 context 有成员变量 strategy 或者其他的策略接口

,至于需要使用到哪个策略,我们可以在构造器中指定

6 策略模式解决鸭子问题

1) 应用实例要求

编写程序完成前面的鸭子项目,要求使用策略模式

2) 思路分析(类图)

策略模式:分别封装行为接口,实现算法族,超类里放行为接口对象,在子类里具体设定行为对象。原则就是: 分离变化部分,封装接口,基于接口编程各种功能。此模式让行为的变化独立于算法的使用者

3)代码实现

  1. package com.lin.strategy.plus;
  2.  
  3. public abstract class Duck {
  4.  
  5. // 策略接口
  6. public FlyBehavior flyBehavior;
  7.  
  8. public abstract void display();
  9.  
  10. public void quack() {
  11. System.out.println("鸭子嘎嘎嘎嘎");
  12. }
  13.  
  14. public void swimming() {
  15. System.out.println("鸭子会游泳");
  16. }
  17.  
  18. public void fly() {
  19. if(flyBehavior != null) {
  20. flyBehavior.fly();
  21. }
  22. }
  23.  
  24. // 动态改变某个对象的行为
  25. public void setFly(FlyBehavior flyBehavior) {
  26. this.flyBehavior = flyBehavior;
  27. }
  28. }
  1. package com.lin.strategy.plus;
  2.  
  3. public class WildDuck extends Duck{
  4.  
  5. public WildDuck() {
  6. super.flyBehavior = new GoodFly();
  7. }
  8.  
  9. @Override
  10. public void display() {
  11. System.out.println("这是野鸭!");
  12. }
  13.  
  14. }
  1. package com.lin.strategy.plus;
  2.  
  3. public class ToyDuck extends Duck {
  4.  
  5. public ToyDuck() {
  6. flyBehavior = new NotFly();
  7. }
  8.  
  9. @Override
  10. public void display() {
  11. System.out.println("玩具鸭");
  12.  
  13. }
  14.  
  15. // 要重写所有父类的方法
  16. public void quack() {
  17. System.out.println("鸭子不会嘎嘎嘎嘎");
  18. }
  19.  
  20. public void swimming() {
  21. System.out.println("鸭子不会游泳");
  22. }
  23.  
  24. }
  1. package com.lin.strategy.plus;
  2.  
  3. public class PekingDuck extends Duck {
  4.  
  5. public PekingDuck() {
  6. flyBehavior = new NotFly();
  7. }
  8.  
  9. @Override
  10. public void display() {
  11. System.out.println("北京鸭!");
  12.  
  13. }
  14.  
  15. }
  1. package com.lin.strategy.plus;
  2.  
  3. public class GoodFly implements FlyBehavior{
  4.  
  5. @Override
  6. public void fly() {
  7. System.out.println("飞翔技术十分好");
  8. }
  9.  
  10. }
  11.  
  12. class NotFly implements FlyBehavior{
  13.  
  14. @Override
  15. public void fly() {
  16. System.out.println("不会飞翔");
  17. }
  18.  
  19. }
  20.  
  21. class BadFly implements FlyBehavior{
  22.  
  23. @Override
  24. public void fly() {
  25. System.out.println("飞翔技术很差");
  26. }
  27.  
  28. }
  1. package com.lin.strategy.plus;
  2.  
  3. public interface FlyBehavior {
  4.  
  5. void fly();
  6. }
  1. package com.lin.strategy.plus;
  2.  
  3. public class Client {
  4.  
  5. public static void main(String[] args) {
  6.  
  7. PekingDuck pekingDuck = new PekingDuck();
  8. pekingDuck.fly();
  9.  
  10. // 动态改变某个对象的行为
  11. pekingDuck.setFly(new GoodFly());
  12. pekingDuck.fly();
  13. }
  14.  
  15. }

7 策略模式在 JDK-Arrays 应用的源码分析

1) JDK 的 Arrays 的 Comparator 就使用了策略模式

2)说明:从上图可以看到,客户 context 有成员变量 strategy 或者其他的策略接口

3)代码分析+模式角色分析

  1. package com.lin.strategy.plus;
  2.  
  3. import java.util.Arrays;
  4. import java.util.Comparator;
  5.  
  6. public class StrategyTest {
  7.  
  8. public static void main(String[] args) {
  9.  
  10. // 实现降序排序,返回-1 放左边,1 放右边,0 保持不变
  11. // 说 明
  12. // 1. 实现了 Comparator 接口(策略接口) , 匿名类 对象 new Comparator<Integer>(){..}
  13. // 2. 对象 new Comparator<Integer>(){..} 就是实现了 策略接口 的对象
  14. // 3. public int compare(Integer o1, Integer o2){} 指定具体的处理方式
  15. Integer[] data = {3,4,6,78,1,0,-91};
  16. Comparator<Integer> comparator = new Comparator<Integer>() {
  17.  
  18. @Override
  19. public int compare(Integer o1, Integer o2) {
  20. if(o1<o2) { // 降序,升序
  21. return 1;
  22. } else {
  23. return -1;
  24. }
  25. }
  26. };
  27.  
  28. // 方式一
  29. Arrays.sort(data, comparator);
  30. System.out.println(Arrays.toString(data));
  31.  
  32. // 方式二
  33. Integer[] data1 = {3,4,6,78,1,0,-91};
  34. Arrays.sort(data1, (var1, var2) -> {
  35. if(var1.compareTo(var2) > 0) {
  36. return 1;
  37. } else {
  38. return -1;
  39. }
  40. });
  41.  
  42. System.out.println(Arrays.toString(data1));
  43. }
  44. }

8 策略模式的注意事项和细节

1) 策略模式的关键是:分析项目中变化部分与不变部分

2) 策略模式的核心思想是:多用组合/聚合 少用继承;用行为类组合,而不是行为的继承。更有弹性

3) 体现了“对修改关闭,对扩展开放”原则,客户端增加行为不用修改原有代码,只要添加一种策略(或者行为) 即可,避免了使用多重转移语句(if..else if..else)

4) 提供了可以替换继承关系的办法: 策略模式将算法封装在独立的 Strategy 类中使得你可以独立于其 Context 改变它,使它易于切换、易于理解、易于扩展

5) 需要注意的是:每添加一个策略就要增加一个类,当策略过多是会导致类数目庞,至于需要使用到哪个策略,我们可以在构造器中指定

仅供参考,有错误还请指出!

有什么想法,评论区留言,互相指教指教。

觉得不错的可以点一下右边的推荐哟!

祝大家牛年大吉大利,牛气冲天!

设计模式(二十三)——策略模式(Arrays源码分析)的更多相关文章

  1. Java的三种代理模式&完整源码分析

    Java的三种代理模式&完整源码分析 参考资料: 博客园-Java的三种代理模式 简书-JDK动态代理-超详细源码分析 [博客园-WeakCache缓存的实现机制](https://www.c ...

  2. 设计模式(九)——装饰者模式(io源码分析)

    1 星巴克咖啡订单项目(咖啡馆): 1) 咖啡种类/单品咖啡:Espresso(意大利浓咖啡).ShortBlack.LongBlack(美式咖啡).Decaf(无因咖啡) 2) 调料:Milk.So ...

  3. dubbo负载均衡策略及对应源码分析

    在集群负载均衡时,Dubbo 提供了多种均衡策略,缺省为 random 随机调用.我们还可以扩展自己的负责均衡策略,前提是你已经从一个小白变成了大牛,嘻嘻 1.Random LoadBalance 1 ...

  4. Seata AT 模式启动源码分析

    从上一篇文章「分布式事务中间件Seata的设计原理」讲了下 Seata AT 模式的一些设计原理,从中也知道了 AT 模式的三个角色(RM.TM.TC),接下来我会更新 Seata 源码分析系列文章. ...

  5. Eclipse用法和技巧二十三:查看JDK源码

    使用java开发,如果能阅读JDK的经典代码,对自己的水平提高是很有帮助的.笔者在实际工作中总结了两种阅读JDK源码的方式.第一种下载android源代码,直接在android源码代码中,这里的代码虽 ...

  6. Java I/O系列(二)ByteArrayInputStream与ByteArrayOutputStream源码分析及理解

    1. ByteArrayInputStream 定义 继承了InputStream,数据源是内置的byte数组buf,那read ()方法的使命(读取一个个字节出来),在ByteArrayInputS ...

  7. Python策略模式实现源码分享

    1.让一个对象的某个方法可以随时改变,而不用更改对象的代码 2.对于动态类型的Python语言,不需要定义接口 3.基本的实现方法:用类作为参数传递 例如: 12_eg3.py class Movea ...

  8. Future模式的学习以及JDK内置Future模式的源码分析

    并发程序设计之Future模式 一).使用Future模式的原因 当某一段程序提交了一个请求,期待得到一个答复,但服务程序对这个请求的处理可能很慢,在单线程的环境中,调用函数是同步的,必须等到服务程序 ...

  9. epoll的ET和LT模式比较 - 源码分析

    eventpoll是一种文件,它实现了一种机制利用一条rdllist队列来避免阻塞地进行poll.eventpoll归根到底还是在使用poll.而ET比LT高效,并不在于是否使用了poll,更不能说是 ...

随机推荐

  1. 迈凯伦765LT/600LT/720S/650S/570S维修手册电路图Mclaren车间手册接线图

    全套迈凯伦维修手册电路图Mclaren车间手册线路图:语言:English,German,French,Spanish,Chinese,Japanese.McLaren迈凯伦新GT维修手册电路图零配件 ...

  2. 如何创建一个Java项目

    目录 新建项目 项目信息配置 创建Java类 编译和运行 新建项目 首先双击eclipse进入到eclipse页面. 菜单"File"下的"New"里" ...

  3. 使用Bat自动打包并通过FTP发送到备份服务器——实战测试

    这个bat文件要求本地安装有winrar解压软件,位置是:C:\Program Files\WinRAR\WinRAR.exe 如果执行报错,请检查你复制我的代码是否有问题,有些复制粘贴进去后因为一些 ...

  4. navicat premium 11.0.17 破解版

    下载地址: 链接:https://pan.baidu.com/s/1zBoKRAaQZb2p2weelJpKMQ       提取码:b8dd 一款功能强大的数据库管理工具Navicat Premiu ...

  5. hook笔记②

  6. 大型 web 前端架构设计-面向抽象编程入门

    https://mp.weixin.qq.com/s/GG6AtBz6KgNwplpaNXfggQ 大型 web 前端架构设计-面向抽象编程入门 曾探 腾讯技术工程 2021-01-04   依赖反转 ...

  7. 进程通信类型 管道是Linux支持的最初Unix IPC形式之一 命名管道 匿名管道

    管道 Linux环境进程间通信(一) https://www.ibm.com/developerworks/cn/linux/l-ipc/part1/index.html 管道及有名管道 郑彦兴200 ...

  8. Excel 一张表最多能装下多少行多少列数据?

    一个工作簿可以装下255张,那么每张工作表可以装下多少行多少列数据呢? 1.任意打开或新建一个Excel文档. 2.在文档中,找到其左上角的"文件"按钮,点击选择"选项& ...

  9. jQuery——样式与动画

    通过jQuery,不仅能够轻松地为页面操作添加简单的视觉效果,甚至能创建更精致的动画. ###修改内联CSS jQuery提供了.css()方法. 这个方法集getter(获取方法)和setter(设 ...

  10. SLAM01

    上周末发现了一个巨大的问题,就是我们目前构建的室内定位的方法中,一个基本的假设是错的----这就非常尴尬了. 于是乎赶紧抱一波佛脚,学习一下slam里相关的问题是怎么解决的,找找灵感. 结果看了个开头 ...