策略模式-Strategy

策略模式中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为型模式。简单理解就是一组算法,可以互换,再简单点策略就是封装算法。

一、类图

策略模式包含如下三个角色:
  • 环境(Context)角色:持有一个Strategy的引用。
  • 抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。
  • 具体策略(ConcreteStrategy)角色:包装了相关的算法或行为。

二、示例

假设某人在开车时频繁加速,因为超速他被一个警察拦下来了。有可能这个警察会比较友好,没开任何罚单就让他把车开走了。也有可能遇到了一个不太友好的警察,然后这个警察给他出具了一张罚单。但是并不知道会遇到什么类型的警察,直到他因为超速而被警察拦下停车,实际上这就是一种程序当中“运行时”的概念,只有在运行的时候才知道,到底会遇到什么类型的警察,实际上这就是“策略模式”。

策略接口:

//先来定义一个策略的接口:Strategy
public interface Strategy {

    public void processSpeeding(int speed);

}

两种不同类型的Strategy:

public class NicePolice implements Strategy {

    @Override
    public void processSpeeding(int speed) {
        System.out.println("This is your first time, be sure don't do it again!");
    }

}

public class HardPolice implements Strategy {

        @Override
        public void processSpeeding(int speed) {
             System.out.println("Your speed is " + speed+ ", and should get a ticket!");
    }

}

定义需要依赖警察来处理超速问题的场景

public class Situation {

   private Strategy strategy;
    
   public Situation(Strategy strategy){
       this.strategy = strategy;
   }
    
   public void handleByPolice(int speed){
       this.strategy.processSpeeding(speed);
   }

}

测试

public class Main {

    public static void main(String[] args) {
        HardPolice hardpolice = new HardPolice();
        NicePolice nicepolice = new NicePolice();

        // In situation 1, a hard officer is met
        // In situation 2, a nice officer is met
        Situation s1 = new Situation(hardpolice);
        Situation s2 = new Situation(nidepolice);

        // the result based on the kind of police officer.
        s1.handleByPolice(10);
        s2.handleByPolice(10);

    }

}

三、策略模式优缺点

优点:

  • 算法可以自由切换:改一下策略很方便

  • 扩展性良好:增加一个策略,就多增加一个类就好了。

缺点:

  • 策略类的数量增多:每一个策略都是一个类,复用的可能性很小、类数量增多

  • 所有的策略类都需要对外暴露:上层模块必须知道有哪些策略,然后才能决定使用哪一个策略

四、JDK中的策略模式

在创建线程池(ThreadPoolExecutor)时,需要传入拒绝策略,当创建新线程使当前运行的线程数超过maximumPoolSize时,将会使用传入的拒绝策略进行处理。

  • AbortPolicy:直接抛出异常。
  • CallerRunsPolicy:只用调用者所在线程来运行任务。
  • DiscardOldestPolicy:丢弃队列里最近的一个任务,并执行当前任务。
  • DiscardPolicy:不处理,丢弃掉。

这里使用的就是策略模式

 

……更多设计模式的内容,可以访问Refactoring.Guru

posted @ 2019-10-29 18:41  静水楼台/Java部落阁  阅读(602)  评论(0编辑  收藏  举报