精囊妙计 -- 策略模式(Strategy Pattern) 介绍 优缺点 使用场景案例及代码演示

一句话概括:

一个类的行为或算法可以在运行时更改。

补充介绍:

策略模式(Strategy Pattern)中,一个类的行为或其算法可以在运行时更改,这种类型的设计模式属于行为型模式。

Strategy的意思是“策略”,指的是与敌军对垒时行军作战的方法。

在策略模式中,我们创建各种表示策略的对象一个行为随着策略对象改变而改变的context对象。策略对象改变context对象的执行算法。

定义一系列的算法,把策略对象一个个封装起来, 并且使它们可相互替换。

主要解决在有多种算法相似的情况下,使用 if...else 所带来的复杂和难以维护。

参与角色:

1)策略类的抽象基类(也可以是接口)拥有不同策略类的共有方法

2)各种策略实现类

3)Context类(持有策略基类的对象,可以动态随着策略对象的改变而改变行为)

优点:

1)算法可以自由切换。

2)避免使用多重条件判断。

3)扩展性良好。

缺点:

1)策略类会增多。

2)所有策略类都需要对外暴露。

使用案例或场景:

1) 诸葛亮的锦囊妙计,每一个锦囊就是一个策略。

2)旅行的出游方式,选择骑自行车、坐汽车,每一种旅行方式都是一个策略。

3)JAVA AWT 中的 LayoutManager。

示例程序

需要源码的朋友可以前往github下载:

https://github.com/aharddreamer/chendong/tree/master/design-patterns/demo-code/design-patterns

程序简介:

下面这个程序的功能是让电脑玩猜拳游戏。我们考虑了两种猜拳策略,第一种策略是“如果这局获胜,那么下局也出一样的手势” (VinningStrategy),这是一种稍微有些笨的策略;另一种策略是“根据上一局的手势从概率上计算出下一局的手势” (ProbStrategy)。

示例程序类/接口一览:

Hand 表示猜拳游戏中的“手势”的类(工具)

Strategy 表示猜拳游戏中的策略的类 (策略类的基类)

WinningStrategy 表示“如果这局获胜,那么下局也出一样的手势” 这一策略的类 (具体策略对象1)

ProbStrategy 表示“根据上一局的手势从概率上计算出下一局的手势” 这一策略的类 (具体策略对象2)

Player 表示进行猜拳游戏的类 (Context环境类)

StrategyPatternTest 测试程序的类

 

代码:

public class Hand {
    public static final int SHI_TOU = 0; //表示石头的值
    public static final int JIAN_DAO = 1; //表示石头的值
    public static final int BU = 2; //表示石头的值
    public static final Hand[] hand = {
            new Hand(SHI_TOU), new Hand(JIAN_DAO), new Hand(BU)
    };
    public static final String[] name = {
            "石头", "剪刀", "布"
    };
    private int handValue;
    private Hand (int handValue) {
        this.handValue = handValue;
    }
    public boolean isStrongerThan(Hand hand) {
        return fight(hand) == -1;
    }
    public int fight(Hand hand) {
        if (this == hand) {
            return 0;
        }else if ((this.handValue + 1) % 3 == hand.handValue) {
            return 1;
        }else {
            return -1;
        }
    }
    public String toString() {
        return name[this.handValue];
    }
    public static Hand getHand(int handValue) {
        return hand[handValue];
    }
    public boolean isWeakerThan(Hand hand) {
        return fight(hand) == -1;
    }
}



public interface Strategy {
    Hand nextHand();
    void study(boolean vin);
}


public class WinningStrategy implements Strategy {
    private Random random;
    private boolean won = false;
    private Hand preHand;
    public WinningStrategy(int seed) {
        random = new Random(seed);
    }
    public Hand nextHand() {
        if (!won) {
            preHand = Hand.getHand(random.nextInt(3));
        }
        return preHand;
    }
    public void study(boolean win) {
        won = win;
    }
}

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;
    }

    //策略决定下一句要出的手势
    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 "Player{" +
                "name='" + name + '\'' +
                ", strategy=" + strategy +
                ", wincount=" + wincount +
                ", losecount=" + losecount +
                ", gamecount=" + gamecount +
                '}';
    }
}


public class ProbStrategy implements Strategy {
    private Random random;
    private int preHandValue = 0;
    private int currentHandValue = 0;
    private int[][] history = {
            {1, 1, 1, },
            {1, 1, 1, },
            {1, 1, 1, },
    };
    public ProbStrategy(int seed) {
        random = new Random(seed);
    }
    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;
        }
        preHandValue = currentHandValue;
        currentHandValue = handValue;
        return Hand.getHand(handValue);
    }
    private int getSum(int hv) {
        int sum = 0;
        for (int i = 0; i < 3 ; i++) {
            sum += history[hv][i];
        }
        return sum;
    }
    public void  study(boolean win) {
        if (win) {
            history[preHandValue][currentHandValue] ++;
        }else {
            history[preHandValue][(currentHandValue + 1) % 3] ++;
            history[preHandValue][(currentHandValue + 2) % 3] ++;
        }
    }
}

public class StrategyPatternTest {
    public static void main(String[] args) {
        //可以选择动态确定seed
        /*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]);*/
        int seed1 = 314;
        int seed2 = 15;
        Player player1 = new Player("WinningStrategyPlayer", new WinningStrategy(seed1));
        Player player2 = new Player("ProbStrategyPlayer", new ProbStrategy(seed2));
        for (int i = 0; i < 1000 ; 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: " + nextHand2);
                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());
    }
}

运行结果:

Winner: Player{name='WinningStrategyPlayer', strategy=org.cd.designpatterns.strategy.WinningStrategy@2ff4acd0, wincount=310, losecount=330, gamecount=993}

Winner: 石头

Winner: Player{name='WinningStrategyPlayer', strategy=org.cd.designpatterns.strategy.WinningStrategy@2ff4acd0, wincount=311, losecount=331, gamecount=995}

Even...

Winner: 石头

Even...

Even...

-------(省略若干行)---------

Total Result:

Player{name='WinningStrategyPlayer', strategy=org.cd.designpatterns.strategy.WinningStrategy@2ff4acd0, wincount=312, losecount=332, gamecount=1000}

Player{name='ProbStrategyPlayer', strategy=org.cd.designpatterns.strategy.ProbStrategy@54bedef2, wincount=332, losecount=312, gamecount=1000}

posted @ 2019-04-19 00:17  SEC.VIP_网络安全服务  阅读(180)  评论(0编辑  收藏  举报