设计模式(二十三)——策略模式(Arrays源码分析)
1 编写鸭子项目,具体要求如下:
1) 有各种鸭子(比如 野鸭、北京鸭、水鸭等, 鸭子有各种行为,比如 叫、飞行等)
2) 显示鸭子的信息
2 传统方案解决鸭子问题的分析和代码实现
1) 传统的设计方案(类图)
2)代码实现
- package com.lin.strategy;
- public abstract class Duck {
- public abstract void display();
- public void quack() {
- System.out.println("鸭子嘎嘎嘎嘎");
- }
- public void swimming() {
- System.out.println("鸭子会游泳");
- }
- public void fly() {
- System.out.println("鸭子会飞");
- }
- }
- package com.lin.strategy;
- public class PekingDuck extends Duck {
- @Override
- public void display() {
- System.out.println("这是北京鸭");
- }
- // 北京鸭不好飞翔
- @Override
- public void fly() {
- System.out.println("北京鸭不会飞翔");
- }
- }
- package com.lin.strategy;
- public class ToyDuck extends Duck {
- @Override
- public void display() {
- System.out.println("玩具鸭");
- }
- // 要重写所有父类的方法
- public void quack() {
- System.out.println("鸭子不会嘎嘎嘎嘎");
- }
- public void swimming() {
- System.out.println("鸭子不会游泳");
- }
- public void fly() {
- System.out.println("鸭子不会飞");
- }
- }
- package com.lin.strategy;
- public class WildDuck extends Duck{
- @Override
- public void display() {
- System.out.println("这是野鸭!");
- }
- }
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)代码实现
- package com.lin.strategy.plus;
- public abstract class Duck {
- // 策略接口
- public FlyBehavior flyBehavior;
- public abstract void display();
- public void quack() {
- System.out.println("鸭子嘎嘎嘎嘎");
- }
- public void swimming() {
- System.out.println("鸭子会游泳");
- }
- public void fly() {
- if(flyBehavior != null) {
- flyBehavior.fly();
- }
- }
- // 动态改变某个对象的行为
- public void setFly(FlyBehavior flyBehavior) {
- this.flyBehavior = flyBehavior;
- }
- }
- package com.lin.strategy.plus;
- public class WildDuck extends Duck{
- public WildDuck() {
- super.flyBehavior = new GoodFly();
- }
- @Override
- public void display() {
- System.out.println("这是野鸭!");
- }
- }
- package com.lin.strategy.plus;
- public class ToyDuck extends Duck {
- public ToyDuck() {
- flyBehavior = new NotFly();
- }
- @Override
- public void display() {
- System.out.println("玩具鸭");
- }
- // 要重写所有父类的方法
- public void quack() {
- System.out.println("鸭子不会嘎嘎嘎嘎");
- }
- public void swimming() {
- System.out.println("鸭子不会游泳");
- }
- }
- package com.lin.strategy.plus;
- public class PekingDuck extends Duck {
- public PekingDuck() {
- flyBehavior = new NotFly();
- }
- @Override
- public void display() {
- System.out.println("北京鸭!");
- }
- }
- package com.lin.strategy.plus;
- public class GoodFly implements FlyBehavior{
- @Override
- public void fly() {
- System.out.println("飞翔技术十分好");
- }
- }
- class NotFly implements FlyBehavior{
- @Override
- public void fly() {
- System.out.println("不会飞翔");
- }
- }
- class BadFly implements FlyBehavior{
- @Override
- public void fly() {
- System.out.println("飞翔技术很差");
- }
- }
- package com.lin.strategy.plus;
- public interface FlyBehavior {
- void fly();
- }
- package com.lin.strategy.plus;
- public class Client {
- public static void main(String[] args) {
- PekingDuck pekingDuck = new PekingDuck();
- pekingDuck.fly();
- // 动态改变某个对象的行为
- pekingDuck.setFly(new GoodFly());
- pekingDuck.fly();
- }
- }
7 策略模式在 JDK-Arrays 应用的源码分析
1) JDK 的 Arrays 的 Comparator 就使用了策略模式
2)说明:从上图可以看到,客户 context 有成员变量 strategy 或者其他的策略接口
3)代码分析+模式角色分析
- package com.lin.strategy.plus;
- import java.util.Arrays;
- import java.util.Comparator;
- public class StrategyTest {
- public static void main(String[] args) {
- // 实现降序排序,返回-1 放左边,1 放右边,0 保持不变
- // 说 明
- // 1. 实现了 Comparator 接口(策略接口) , 匿名类 对象 new Comparator<Integer>(){..}
- // 2. 对象 new Comparator<Integer>(){..} 就是实现了 策略接口 的对象
- // 3. public int compare(Integer o1, Integer o2){} 指定具体的处理方式
- Integer[] data = {3,4,6,78,1,0,-91};
- Comparator<Integer> comparator = new Comparator<Integer>() {
- @Override
- public int compare(Integer o1, Integer o2) {
- if(o1<o2) { // 降序,升序
- return 1;
- } else {
- return -1;
- }
- }
- };
- // 方式一
- Arrays.sort(data, comparator);
- System.out.println(Arrays.toString(data));
- // 方式二
- Integer[] data1 = {3,4,6,78,1,0,-91};
- Arrays.sort(data1, (var1, var2) -> {
- if(var1.compareTo(var2) > 0) {
- return 1;
- } else {
- return -1;
- }
- });
- System.out.println(Arrays.toString(data1));
- }
- }
8 策略模式的注意事项和细节
1) 策略模式的关键是:分析项目中变化部分与不变部分
2) 策略模式的核心思想是:多用组合/聚合 少用继承;用行为类组合,而不是行为的继承。更有弹性
3) 体现了“对修改关闭,对扩展开放”原则,客户端增加行为不用修改原有代码,只要添加一种策略(或者行为) 即可,避免了使用多重转移语句(if..else if..else)
4) 提供了可以替换继承关系的办法: 策略模式将算法封装在独立的 Strategy 类中使得你可以独立于其 Context 改变它,使它易于切换、易于理解、易于扩展
5) 需要注意的是:每添加一个策略就要增加一个类,当策略过多是会导致类数目庞,至于需要使用到哪个策略,我们可以在构造器中指定
仅供参考,有错误还请指出!
有什么想法,评论区留言,互相指教指教。
觉得不错的可以点一下右边的推荐哟!
祝大家牛年大吉大利,牛气冲天!
设计模式(二十三)——策略模式(Arrays源码分析)的更多相关文章
- Java的三种代理模式&完整源码分析
Java的三种代理模式&完整源码分析 参考资料: 博客园-Java的三种代理模式 简书-JDK动态代理-超详细源码分析 [博客园-WeakCache缓存的实现机制](https://www.c ...
- 设计模式(九)——装饰者模式(io源码分析)
1 星巴克咖啡订单项目(咖啡馆): 1) 咖啡种类/单品咖啡:Espresso(意大利浓咖啡).ShortBlack.LongBlack(美式咖啡).Decaf(无因咖啡) 2) 调料:Milk.So ...
- dubbo负载均衡策略及对应源码分析
在集群负载均衡时,Dubbo 提供了多种均衡策略,缺省为 random 随机调用.我们还可以扩展自己的负责均衡策略,前提是你已经从一个小白变成了大牛,嘻嘻 1.Random LoadBalance 1 ...
- Seata AT 模式启动源码分析
从上一篇文章「分布式事务中间件Seata的设计原理」讲了下 Seata AT 模式的一些设计原理,从中也知道了 AT 模式的三个角色(RM.TM.TC),接下来我会更新 Seata 源码分析系列文章. ...
- Eclipse用法和技巧二十三:查看JDK源码
使用java开发,如果能阅读JDK的经典代码,对自己的水平提高是很有帮助的.笔者在实际工作中总结了两种阅读JDK源码的方式.第一种下载android源代码,直接在android源码代码中,这里的代码虽 ...
- Java I/O系列(二)ByteArrayInputStream与ByteArrayOutputStream源码分析及理解
1. ByteArrayInputStream 定义 继承了InputStream,数据源是内置的byte数组buf,那read ()方法的使命(读取一个个字节出来),在ByteArrayInputS ...
- Python策略模式实现源码分享
1.让一个对象的某个方法可以随时改变,而不用更改对象的代码 2.对于动态类型的Python语言,不需要定义接口 3.基本的实现方法:用类作为参数传递 例如: 12_eg3.py class Movea ...
- Future模式的学习以及JDK内置Future模式的源码分析
并发程序设计之Future模式 一).使用Future模式的原因 当某一段程序提交了一个请求,期待得到一个答复,但服务程序对这个请求的处理可能很慢,在单线程的环境中,调用函数是同步的,必须等到服务程序 ...
- epoll的ET和LT模式比较 - 源码分析
eventpoll是一种文件,它实现了一种机制利用一条rdllist队列来避免阻塞地进行poll.eventpoll归根到底还是在使用poll.而ET比LT高效,并不在于是否使用了poll,更不能说是 ...
随机推荐
- 迈凯伦765LT/600LT/720S/650S/570S维修手册电路图Mclaren车间手册接线图
全套迈凯伦维修手册电路图Mclaren车间手册线路图:语言:English,German,French,Spanish,Chinese,Japanese.McLaren迈凯伦新GT维修手册电路图零配件 ...
- 如何创建一个Java项目
目录 新建项目 项目信息配置 创建Java类 编译和运行 新建项目 首先双击eclipse进入到eclipse页面. 菜单"File"下的"New"里" ...
- 使用Bat自动打包并通过FTP发送到备份服务器——实战测试
这个bat文件要求本地安装有winrar解压软件,位置是:C:\Program Files\WinRAR\WinRAR.exe 如果执行报错,请检查你复制我的代码是否有问题,有些复制粘贴进去后因为一些 ...
- navicat premium 11.0.17 破解版
下载地址: 链接:https://pan.baidu.com/s/1zBoKRAaQZb2p2weelJpKMQ 提取码:b8dd 一款功能强大的数据库管理工具Navicat Premiu ...
- hook笔记②
- 大型 web 前端架构设计-面向抽象编程入门
https://mp.weixin.qq.com/s/GG6AtBz6KgNwplpaNXfggQ 大型 web 前端架构设计-面向抽象编程入门 曾探 腾讯技术工程 2021-01-04 依赖反转 ...
- 进程通信类型 管道是Linux支持的最初Unix IPC形式之一 命名管道 匿名管道
管道 Linux环境进程间通信(一) https://www.ibm.com/developerworks/cn/linux/l-ipc/part1/index.html 管道及有名管道 郑彦兴200 ...
- Excel 一张表最多能装下多少行多少列数据?
一个工作簿可以装下255张,那么每张工作表可以装下多少行多少列数据呢? 1.任意打开或新建一个Excel文档. 2.在文档中,找到其左上角的"文件"按钮,点击选择"选项& ...
- jQuery——样式与动画
通过jQuery,不仅能够轻松地为页面操作添加简单的视觉效果,甚至能创建更精致的动画. ###修改内联CSS jQuery提供了.css()方法. 这个方法集getter(获取方法)和setter(设 ...
- SLAM01
上周末发现了一个巨大的问题,就是我们目前构建的室内定位的方法中,一个基本的假设是错的----这就非常尴尬了. 于是乎赶紧抱一波佛脚,学习一下slam里相关的问题是怎么解决的,找找灵感. 结果看了个开头 ...