设计模式学习(四):Strategy策略模式

news2025/1/12 22:47:12

一、什么是Strategy模式

        Strategy的意思是“策略”,指的是与敌军对垒时行军作战的方法。在编程中,我们可以将它理解为“算法”。无论什么程序,其目的都是解决问题。而为了解决问题,我们又需要编写特定的算法。使用Strategy模式可以整体地替换算法的实现部分,能让我们轻松地以不同的算法去解决同一个问题,这种模式就是Strategy模式。

        用一句话概况:可以整体地替换算法。

二、Strategy模式示例代码 

        这段示例程序的功能是让电脑玩“猜拳”游戏。 

       我们考虑了两种猜拳的策略。第一种策略是“如果这局猜拳获胜,那么下一局也出一样的手势”( WinningStrategy),这是一种稍微有些笨的策略;另外一种策略是“根据上一局的手势从概率上计算出下一局的手势”( ProbStrategy )。

2.1 各个类之间的关系

        先看一下所有的类和接口: 


 

        再看一下类图:

2.2 Hand类 

        Hand类的实例可以通过使用类方法 getHand来获取。只要将表示手势的值作为参数传递给getHand方法,它就会将手势的值所对应的Hand类的实例返回给我们。这也是一种 Singleton模式。

public class Hand {

    //石头的值为0
    public static final int HANDVALUE_SHITOU = 0;
    //剪刀的值为1
    public static final int HANDVALUE_JIANDAO = 1;
    //布的值为2
    public static final int HANDVALUE_BU = 2;

    //三种手势的实例
    public static final Hand[] hand = {
            new Hand(HANDVALUE_SHITOU),
            new Hand(HANDVALUE_JIANDAO),
            new Hand(HANDVALUE_BU)
    };

    //手势对应的字符串
    private static final String []name = {
            "石头", "剪刀", "布"
    };

    //猜拳中出的手势的值
    private int handValue;

    private Hand(int handValue) {
        this.handValue = handValue;
    }

    //根据手势的值获取对应的实例
    public static Hand getHand(int handValue) {
        return hand[handValue];
    }

    //如果this胜了h返回true
    public boolean isStrongerThan(Hand h) {
        return fight(h)==1;
    }

    //如果this输了h返回true
    public boolean isWeakerThan(Hand h) {
        return fight(h)==-1;
    }

    //实际用来判断胜负的方法:平0分,胜1分,输-1分
    private int fight(Hand h) {
        if (this == h) {
            return 0;
        } else if ((this.handValue+1)%3 == h.handValue) {
            return 1;
        } else {
            return -1;
        }
    }

    public String toString() {
        return name[handValue];
    }
}

2.3 Strategy接口

        定义了猜拳策略的抽象方法的接口。 

public interface Strategy {
    /**
     * 获取下一局要出的手势
     */
    public abstract Hand nextHand();

    /**
     * 学习上一局的手势是否获胜了,为下一次出什么手势提供依据
     * @param win 上一局是否获胜
     */
    public abstract void study(boolean win);
}

2.4 WinningStrategy类 

        实现的猜拳策略 WinningStrategy。

/**
 * 该类的猜拳策略有些笨。如果上一局的手势获胜了,则下一局的手势就与上局相同;如果上一局的手势输了,则下一局就随机出手势。
 */
public class WinningStrategy implements Strategy{

    private Random random;
    //保存了上一局猜拳的输赢结果
    private boolean won = false;
    //上一局出的手势
    private Hand prevHand;

    public WinningStrategy(int seed) {
        random = new Random(seed);
    }

    @Override
    public Hand nextHand() {
        if (!won) {
            prevHand = Hand.getHand(random.nextInt(3));
        }
        return prevHand;
    }

    @Override
    public void study(boolean win) {
        won = win;
    }
}

2.5  ProbStrategy类

        实现的猜拳策略 ProbStrategy。 

public class ProbStrategy implements Strategy{

    private Random random;
    private int prevHandValue = 0;
    private int currentHandValue = 0;

    /**
     * history[上一局出的手势][这一局所出的手势]
     * 这个表达式的值越大,表示过去的胜率越高。下面稍微详细讲解下:
     * 假设我们上一局出的是石头。
     * history[0][0]两局分别出石头、石头时胜了的次数
     * history[0][1]两局分别出石头、剪刀时胜了的次数
     * history[0][2]两局分别出石头、布时胜了的次数
     */
    private int[][] history = {
            {1, 1, 1, },
            {1, 1, 1, },
            {1, 1, 1, },
    };

    public ProbStrategy(int seed) {
        random = new Random(seed);
    }

    /**
     * 那么,我们就可以根据 history[0][0]、history[0][1]、history[0][2]这3个表达式的值从概率上计算出下一局出什么。
     * 简而言之,就是先计算3个表达式的值的和 (getSum方法),然后再从0与这个和之间取一个随机数,并据此决定下一局应该出什么( nextHand方法)。
     * 例如,如果
     * history[0][0]是3
     * history[0][1]是5
     * history[0][2]是7
     * 那么,下一局出什么就会以石头、剪刀和布的比率为3:5:7来决定。然后在0至15(不含15,15是3+5+7的和)之间取一个随机数。
     */
    @Override
    public Hand nextHand() {
        int bet = random.nextInt(getSum(currentHandValue));
        int handvalue = 0;
        if (bet < history[currentHandValue][0]) {
            handvalue = 0;
        } else if (bet < history[currentHandValue][0] + history[currentHandValue][1]) {
            handvalue = 1;
        } else {
            handvalue = 2;
        }
        prevHandValue = currentHandValue;
        currentHandValue = handvalue;
        return Hand.getHand(handvalue);
    }

    /**
     * study方法会根据nextHand方法返回的手势的胜负结果来更新history字段中的值。
     * @param win 上一局是否获胜
     */
    @Override
    public void study(boolean win) {
        if (win) {
            history[prevHandValue][currentHandValue]++;
        } else {
            history[prevHandValue][(currentHandValue+1)%3]++;
            history[prevHandValue][(currentHandValue+2)%3]++;
        }
    }

    private int getSum(int hv) {
        int sum = 0;
        for (int i = 0; i < 3; i++) {
            sum += history[hv][i];
        }
        return sum;
    }
}

 2.6 Play类

        Player类是表示进行猜拳游戏的选手的类。

        nextHand方法是用来获取下一局手势的方法,不过实际上决定下一局手势的是各个策略。Player类的nextHand方法的返回值其实就是策略的nextHand方法的返回值。nextHand方法将自己的工作委托给了strategy,这就形成了一种委托关系。

        在决定下一局要出的手势时,需要知道之前各局的胜(win)、负(lose)、平(even)等结果,因此Player类会通过strategy字段调用study方法,然后study方法会改变策略的内部状态。wincount、losecount 和 gamecount用于记录选手的猜拳结果。

public class Player {
    //选手姓名
    private String name;
    //选手所选策略
    private Strategy strategy;

    //选手猜拳结果
    private int wincount;
    private int losecount;
    private int gamecount;

    public Player(String name, Strategy strategy) {
        this.name = name;
        this.strategy = strategy;
    }

    //获取下一局手势,实际上决定下一局手势的是各个策略,nextHand方法仅仅是获取
    public Hand nextHand() {
        return strategy.nextHand();
    }

    //胜
    public void win() {
        strategy.study(true);
        wincount++;
        gamecount++;
    }

    //负
    public void lose() {
        strategy.study(false);
        losecount++;
        gamecount++;
    }

    //平
    public void even() {
        gamecount++;
    }

    @Override
    public String toString() {
        return "[" + name + ":" + gamecount + "games," + wincount + "win," + losecount + "lose" + "]";
    }
}

2.7 用于测试的Main类 

       这里Main类让以下两位选手进行10 000局比赛,然后显示比赛结果:

  •         姓名:"Taro"、策略:WinningStrategy
  •         姓名: "Hana"、策略:ProbStrategy
public class Main {
    public static void main(String[] args) {
        if (args.length != 2) {
            System.out.println("Usage: java Main randomseed1 randomseed2");
            System.out.println("Example: java Main 314 15");
            System.exit(0);
        }
        int seed1 = Integer.parseInt(args[0]);
        int seed2 = Integer.parseInt(args[1]);
        Player player1 = new Player("Taro", new WinningStrategy(seed1));
        Player player2 = new Player("Hana", new ProbStrategy(seed2));
        for (int i = 0; i < 10000; i++) {
            Hand nextHand1 = player1.nextHand();
            Hand nextHand2 = player2.nextHand();
            if (nextHand1.isStrongerThan(nextHand2)) {
                System.out.println("Winner:" + player1);
                player1.win();
                player2.lose();
            } else if (nextHand2.isStrongerThan(nextHand1)) {
                System.out.println("Winner:" + player2);
                player1.lose();
                player2.win();
            } else {
                System.out.println("Even...");
                player1.even();
                player2.even();
            }
        }
        System.out.println("Total result:");
        System.out.println(player1.toString());
        System.out.println(player2.toString());
    }
}

三、拓展思路的要点

3.1 为什么需要特意编写Strategy 角色 

        通常在编程时算法会被写在具体方法中。Strategy模式却特意将算法与其他部分分离开来,只是定义了与算法相关的接口(API ),然后在程序中以委托的方式来使用算法。

        这样看起来程序好像变复杂了,其实不然。例如,当我们想要通过改善算法来提高算法的处理速度时,如果使用了Strategy模式,就不必修改Strategy角色的接口(API)了,仅仅修改ConcreteStrategy 角色即可。

        而且,使用委托这种弱关联关系可以很方便地整体替换算法。例如,如果想比较原来的算法与改进后的算法的处理速度有多大区别,简单地替换下算法即可进行测试。

        例如,使用Strategy模式编写象棋程序时,可以方便地根据棋手的选择切换AI例程的水平。
 

3.2 程序运行中也可以切换策略 

        如果使用Strategy模式,在程序运行中也可以切换ConcreteStrategy 角色。例如,在内存容量少的运行环境中可以使用slowButLessMemorystrategy(速度慢但省内存的策略),而在内存容量多的运行环境中则可以使用FastButMoreMemorystrategy(速度快但耗内存的策略)。

        此外,还可以用某种算法去“验算”另外一种算法。例如,假设要在某个表格计算软件的开发版本中进行复杂的计算。这时,我们可以准备两种算法,即“高速但计算上可能有Bug的算法”和“低速但计算准确的算法”,然后让后者去验算前者的计算结果。

四、相关的设计模式 

4.1 Flyweight模式

        有时会使用Flyweight模式让多个地方可以共用ConcreteStrategy角色。

4.2 Abstract Factory模式

        使用Strategy模式可以整体地替换算法。

        使用Abstract Factory模式则可以整体地替换具体工厂、零件和产品。

4.3 State模式

        使用Strategy模式和State模式都可以替换被委托对象,而且它们的类之间的关系也很相似。但是两种模式的目的不同。

        在Strategy模式中,ConcreteStrategy 角色是表示算法的类。在Strategy模式中,可以替换被委托对象的类。当然如果没有必要,也可以不替换。

        而在State模式中,ConcreteState角色是表示“状态”的类。在State模式中,每次状态变化时,被委托对象的类都必定会被替换。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/156578.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Redis- 主从复制原理

1、概述 Master节点在平时提供服务&#xff0c;另外一个或多个Slave节点在平时不提供服务&#xff08;或只提供数据读取服务&#xff09;。当Master节点由于某些原因停止服务后&#xff0c;再人工/自动完成Slave节点到Master节点的切换工作&#xff0c;以便整个Redis集群继续向…

Spring依赖注入源码分析

1. 前言 Spring的核心之一就是依赖注入&#xff0c;Spring提供了Autowired注解来给bean注入依赖。除了注入最基本的bean之外&#xff0c;Spring还做了一些扩展&#xff0c;例如你可以注入Optional&#xff0c;以此来判断依赖的bean是否存在&#xff1b;你还可以注入Map来获得所…

Leetcode:617. 合并二叉树(C++)

目录 问题描述&#xff1a; 实现代码与解析&#xff1a; 递归&#xff1a; 原理思路&#xff1a; 迭代&#xff1a; 原理思路&#xff1a; 问题描述&#xff1a; 给你两棵二叉树&#xff1a; root1 和 root2 。 想象一下&#xff0c;当你将其中一棵覆盖到另一棵之上时&am…

leetcode 399. 除法求值-java题解

题目所属分类 flod最短路算法 原题链接 给你一个变量对数组 equations 和一个实数值数组 values 作为已知条件&#xff0c;其中 equations[i] [Ai, Bi] 和 values[i] 共同表示等式 Ai / Bi values[i] 。每个 Ai 或 Bi 是一个表示单个变量的字符串。 另有一些以数组 queri…

编译metabase

Linux Centos7 配置Metabase编译打包环境 安装Oracle JDK1.8&#xff08;如果已经安装&#xff0c;则可以省略此步骤&#xff0c;必须是Oracle JDK&#xff09; 在线下载Oracle JDK 1.8 将下载好的tar包放入linux目录下 2、解压tar进行安装 tar -zxvf jdk-8u212-linux-x64.t…

SSL/TLS协议信息泄露漏洞(CVE-2016-2183)

最近服务器扫描出SSL/TLS协议信息泄露漏洞(CVE-2016-2183) TLS是安全传输层协议&#xff0c;用于在两个通信应用程序之间提供保密性和数据完整性。 TLS, SSH, IPSec协商及其他产品中使用的DES及Triple DES密码存在大约四十亿块的生日界&#xff0c;这可使远程攻击者通过Sweet…

总结几个常用的Git命令的使用方法

目录 1、Git的使用越来越广泛 2、设置Git的用户名和密码并查看 3、建立自己的 Git 仓库 4、将自己的代码提交到远程 (origin) 仓库 5、同步远程仓库的更新到本地仓库 6、分支管理 7、获取远程仓库的内容 1、Git的使用越来越广泛 现在很多的公司或者机构都在使用Git进行项目和代…

Elasticsearch基础1——搜索引擎发展史和工作流程、es\es-head\kibana的基础安装

文章目录一、搜索引擎1.1 搜索引擎的发展背景1.2 Lucene和Elasticsearch1.3 Solr和Elasticsearch对比1.4 数据搜索方式1.5 搜索引擎1.5.1 搜索引擎工作流程1.5.2 网络爬虫原理流程1.5.3 网页分析1.5.4 正排索引和倒排索引二、Elasticsearch基础安装1.2 概述简介2.2 安装2.2.1 W…

tensorflow算子注册以及op详解

在自定义的算子时&#xff0c;经常遇到一些函数和宏&#xff0c;这里介绍一下常见的函数和宏 REGISTER_OP 首先我们来思考REGISTER_OP的作用是什么&#xff1f;当我们定义一个tensorflow的算子&#xff0c;首先我们需要tensorflow知道这个算子&#xff0c;也就是说我们要把这…

WeLink的使用

我这里是注册的企业端 流程>手机号验证码 注册成功后登陆 进入首页面 按操作逐步完成信息需求 因个体使用情况不同 在角色分类和组织架构中可根据自己部门或单位的分工分类 【拉人】&#xff1a; 三种方式 主要就是网址超链接和企业码 前提需要用户先注册 【加入审核】是根…

Nginx——反向代理解决跨域问题(Windows)

这个破玩意是真麻烦&#xff0c;必须写一篇文章避避坑了。一、先看看大佬的解释&#xff0c;了解反向代理和跨域问题吧&#xff1a;Nginx反向代理什么是跨域问题二、OK&#xff0c;直接开工&#xff0c;装Nginx下载地址: http://nginx.org/en/download.html如图所示, 选择相应的…

Flink多流转换(Flink Stream Unoin、Flink Stream Connect、Flink Stream Window Join)

文章目录多流转换1、分流操作1.1、在flink 1.13版本中已弃用.split()进行分流1.2、使用&#xff08;process function&#xff09;的侧输出流&#xff08;side output&#xff09;进行分流2、基本合流操作2.1、联合&#xff08;Flink Stream Union&#xff09;2.2、连接&#x…

【Go】实操使用go连接clickhouse

前言 近段时间业务在一个局点测试clickhouse&#xff0c;用java写的代码在环境上一直连接不上clickhouse服务&#xff0c;报错信息也比较奇怪&#xff0c;No client available&#xff0c;研发查了一段时间没查出来&#xff0c;让运维这边继续查&#xff1a; 运维同学查了各种…

OAuth 2.0授权框架详解

简介 在现代的网站中&#xff0c;我们经常会遇到使用OAuth授权的情况&#xff0c;比如有一个比较小众的网站&#xff0c;需要用户登录&#xff0c;但是直接让用户注册就显得非常麻烦&#xff0c;用户可能因为这个原因而流失&#xff0c;那么该网站可以使用OAuth授权&#xff0…

FactoryBean和BeanFactory的区别

1. 前言 “BeanFactory和FactoryBean的区别是什么&#xff1f;&#xff1f;&#xff1f;” 这是Spring非常高频的一道面试题&#xff0c;BeanFactory是Spring bean容器的顶级接口&#xff0c;负责创建和维护容器内所有的bean对象。而FactoryBean是用来创建一类bean的接口&…

数字新基建之数据云

自2021年“新基建”概念火爆以来&#xff0c;相关的政策和技术都不断跟进和发展&#xff0c;由于“新基建”本质上是基础设施向数字化、智能化、网络化方向发展&#xff0c;因此更多的科技领域从业者和投资者都将其称为“数字新基建”。而数据库、数据仓库、大数据平台和数据云…

C语言:整数的存储方式

整数的存储方式 char类型在存储时是按照ASCII码值进行存储&#xff0c;存储方式与整型一致 有符号数与无符号数 char一个字节signed charunsigned char int四个字节signed intunsigned int 各种类型数据均分为有符号和无符号类型&#xff0c;当定义一个int类型或char类型的数…

备库为什么会延迟好几个小时?

在上一篇文章中,我和你介绍了几种可能导致备库延迟的原因。你会发现,这些场景里,不论是偶发性的查询压力,还是备份,对备库延迟的影响一般是分钟级的,而且在备库恢复正常以后都能够追上来。 但是,如果备库执行日志的速度持续低于主库生成日志的速度,那这个延迟就有可能…

百度搜索留痕推广资源整理如何收录排名的?

每日分享&#xff1a;百度对图文类内容的优质标准 &#xff08;1&#xff09;文字的字体、字号与间距需要适配网页&#xff0c;文档分段合理&#xff0c;结构有序&#xff0c;阅读体验舒适。 &#xff08;2&#xff09;在文章中使用小标题准确概括段意&#xff0c;通过加粗、…

vue3 setup语法糖父子组件传值,让女友看得明明白白

前言 最近在想做个cloud项目,gitee上找了个模板项目&#xff0c;前端使用到vue3 typeScript&#xff0c;最近使用到vue3 的父子组件之间的传值&#xff0c;顺便学习一下&#xff0c;在此总结一下&#xff0c;若有不足之处&#xff0c;望大佬们可以指出。 vue3官网&#xff1a…