设计模式 - 策略模式

什么是策略模式

直接看一个场景:开发一个运动会计分软件,其中一个需求就是可以根据用户的选择,是直接计算所有成绩的平均值,还是去掉最大最小然后计算平均值,或者采用其他什么算法计算。

在这个问题中计算平均值的算法是不确定的,会根据需要变化。所以关键点就是如何分割变化。也就是如何从将类中经常变化的部分从类中分离开。如果有这个需求就可以试试看能不能使用策略模式。

模式要素

策略模式包含三个要素:

  • 策略
  • 上下文
  • 具体策略

策略就是一个接口,指定具体策略返回值类型和输入参数。

具体策略就是策略的具体实现,可能包含很多的实现。

上下文其实就是所有方法的一个应用端口。需要包含两个方法:一个是指定具体的策略是哪一个的方法,另一个就是使用策略的方法。

策略模式的优点

增加新的具体策略的时候,不需要修改上下文的代码。

适用的场景

  1. 一个类中定义了多种行为,并且这些行为的出现需要有多个条件语句判断,那么可以使用策略模式避免过多的判断。
  2. 程序的主要类(上下文)不希望包含复杂、与算法相关的复杂数据结构,可以使用策略模式封装算法,和主要类分离。
  3. 需要切换不同的算法。

具体的例子

还是上面提到的例子
策略:(一个接口)

public interface Strategy {
    public double computeAverage(double [] a);
}

具体策略:(多个具体实现)

public class Strategy1 implements Strategy{
    @Override
    public double computeAverage(double[] a) {
        System.out.println("使用第一种方法计算平均值:");
        double score = 0, sum = 0;
        for (int i=0; i<a.length; i++) {
            sum = sum + a[i];
        }
        score = sum/a.length;
        return score;
    }
}


import java.util.Arrays;

public class Strategy2 implements Strategy{
    @Override
    public double computeAverage(double[] a) {
        System.out.println("使用第二种方法计算平均值:");
        if(a.length <= 2)
            return 0;
        double score=0, sum=0;
        Arrays.sort(a);
        for (int i=1; i<a.length-1; i++) {
            sum += a[i];
        }
        score = sum/(a.length-2);
        return score;
    }
}

上下文:(具体实现的同一操作面板)

public class AverageScore {
    Strategy strategy;
    public void setStrategy(Strategy strategy){
        this.strategy = strategy;
    }

    public double getAverage(double [] a){
        if(strategy != null)
            return strategy.computeAverage(a);
        else {
            System.out.println("没有求平均值算法,得到的 -1 不代表平均值!");
            return -1;
        }
    }
}

具体应用

public class lianxi {
    public static void main(String args[]){
        Person zhangsan = new Person();
        zhangsan.setName("zhangsan");

        // 计算两种得分
        double []a = {1, 2, 3, 4, 5};
        AverageScore average_score = new AverageScore();
        average_score.setStrategy(new Strategy1());
        double score1 =  average_score.getAverage(a);
        zhangsan.setScore(score1);
        System.out.println(zhangsan);

        average_score.setStrategy(new Strategy2());
        double score2 = average_score.getAverage(a);
        zhangsan.setScore(score2);
        System.out.println(zhangsan);

    }
}

class Person {
    private double score = 0;
    private String name = "未知姓名";

    public void setName(String name) {
        this.name = name;
    }

    public void setScore(double score) {
        this.score = score;
    }

    @Override
    public String toString() {
        return this.name+"的最后得分:"+this.score;
    }
}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 策略模式 定义一组算法,将每个算法都封装起来,并且使它们之间可以互换。将共同的行为封装成策略接口,不同策略实现类只...
    AnimoBlog阅读 270评论 0 1
  • 模式定义 定义一系列算法,将它们一个个封装起来,并且使他们之间可以相互替换。本模式使得算法可以独立于使用它的客户而...
    FX_SKY阅读 254评论 0 0
  • 概念及定义 概念在完成某一功能时,有时需要根据不同环境采取不同的策略或行为。将这些不同的策略或行为(称为算法)一一...
    maxwellyue阅读 573评论 0 0
  • 一件事实是一条没有性别的真理。 — 纪伯伦 写在前面 策略模式的定义:定义一系列的算法,把每一个算法封装起来,并且...
    Chase_stars阅读 266评论 0 1
  • 定义:定义一组算法,将每个算法都封装起来,并且使它们之间可以互换 策略模式通用类图: Context 封装角色:也...
    小杰的快乐时光阅读 212评论 0 0