策略模式用于算法的自由切换和扩展,实现了算法定义和算法分离的使用
要完成一项任务,可以有多种不同的方式,例如人们外出旅游时可以选择多种不同的出行方式,如自行车、坐汽车、坐高铁或乘飞机等,每一种方式称为一个策略,我们可以根据环境或者条件的不同选择不同的策略来完成该任务。
在实际的软件开发中,一项功能也有很多算法可以实现,如果我们直接把多种算法集中在一个类,或者说使用条件判断语句来进行选择,无疑会增加代码复杂性,不利于维护。
为了解决这些问题,可以定义一些独立的类来封装不同的算法,每一类封装一个具体的算法,将每一个封装算法的类称之为策略(Strategy)。为了保证策略的一致性,一般会用一个抽象的策略类来做算法的定义,每种具体算法对应于一个具体策略类。
策略模式(Strategy Pattern)定义一系列算法,将每个算法封装起来,并让它们可以相互替换。策略模式让算法独立于使用它的客户而变化,也称政策模式(Policy)。策略模式是一种对象行为模式。
Define a famliy of algorithms, encapsulate each one,and make them interchangeable. Strategy lets the algorithm vary independently from clients that use it.
说到策略模式就不得不提一下状态模式了,一个是对状态的封装,另一个则是对算法的封装,因此实现方式上两者有许多共同点,理解起来并不复杂
点这里了解状态模式
策略模式是对算法的封装,把算法的责任和算法本身分割开来,委托给不同对象管理。策略模式通常把一系列算法封装到一系列的策略类里面,作为一个抽象策略类的子类。
由此可以得出结构类图如下:
环境类(Context)是需要使用算法的对象,它在解决某个问题时可以采用多种策略,在环境类中维护一个对抽象策略类的引用实例,用于定义所采用的策略。假如我们不使用策略模式,可能会存在如下代码:
public class Context {
...
public void algorithm(String type) {
...
if(type == "strategyA") {
// 算法A
} else if(type == "strategyB") {
// 算法B
} else if(type == "strategyC") {
// 算法C
}
...
}
...
}
客户端要调用 Context 类的 algorithm() 方法时,需要根据所传入的参数来选择具体算法,这将导致代码过于庞大,不利于维护。而且最大的问题是,在环境类中定义算法,如果需要修改或增加算法,则势必要修改源代码,违反了开闭原则。
导致以上问题的主要原因在于环境类职责过重,即违背了单一职责原则,策略模式正是解决这个问题的好帮手。引入一个抽象策略类(Strategy),在其中定义抽象算法,每一个继承它的具体策略类(ConcreteStrategy)使用具体算法实现某个业务办理。环境类只针对抽象策略类编程,符合依赖倒转原则,出现新的算法时,只需要增加一个新的实现了抽象策略类的具体策略类即可。
public abstract class AbstractStrategy {
public abstract void algorithm();
}
将每一种具体算法作为该抽象策略类的子类
public class ConcreteStrategyA extends AbstractStrategy {
public void algorithm() {
// 算法A
}
}
对于环境类,在它与抽象策略类之间建立一个关联关系
public class Context {
private AbstractStrategy strategy;
public void setStrategy(AbstractStrategy strategy) {
this.strategy = strategy;
}
public void algorithm() {
strategy.algorithm();
}
}
对于用户而言,使用环境类时只需注入一个具体策略对象即可。用户可以灵活更换具体策略类,比如使用配置文件,增加新的具体策略类也很方便,因此策略模式相当于“即插即用的算法”。
需要注意的是,策略模式并不负责“哪一个具体策略类适用于哪一种情况”这个决定,换言之,应当由客户决定在什么情况下使用什么具体策略角色,一定程度上提高了系统灵活性,但前提是客户需要理解所以具体策略类之间的区别及最佳使用场景。
对比状态模式,状态模式可以实现自主的状态切换,并根据状态的不同做出不同的行为。而
策略模式的优点:
策略模式的缺点:
由于结构的相似性,所以很容易会把策略模式和状态模式混肴,但它们是为解决不同的问题而设计的,是完全不同的两种模式。如何区分是策略模式还是状态模式?其区别如下: