常言道:“条条大路通罗马”。在生活中达到某种目标,会有很多种方法。例如你想成为有钱人,你可以傍富婆,你也可以自己努力成为富一代。老规矩,系统来源于生活。在系统中处理某个事件会有多种方法。最简单的,酒店对于不同等级的客户,会有不同的优惠。在传统编码上,我们可以使用if--else解决面对不同的情况以不同的策略解决。但是这不易于维护。这时我们可以想想是否适合策略模式
策略(Strategy)模式的定义:该模式定义了一系列算法,并将每个算法封装起来,使它们可以相互替换,且算法的变化不会影响使用算法的客户。策略模式属于对象行为模式,它通过对算法进行封装,把使用算法的责任和算法的实现分割开来,并委派给不同的对象对这些算法进行管理。
优点:
- 多重条件语句不易维护,而使用策略模式可以避免使用多重条件语句,如 if...else 语句、switch...case 语句。
- 策略模式提供了一系列的可供重用的算法族,恰当使用继承可以把算法族的公共代码转移到父类里面,从而避免重复的代码。
- 策略模式可以提供相同行为的不同实现,客户可以根据不同时间或空间要求选择不同的策略。
- 策略模式提供了对开闭原则的完美支持,可以在不修改原代码的情况下,灵活增加新算法。策略模式把算法的使用放到环境类中,而算法的实现移到具体策略类中,实现了二者的分离。
缺点:
- 客户端必须理解所有策略算法的区别,以便适时选择恰当的算法类。
- 策略模式造成很多的策略类,增加维护难度。
策略模式的结构与实现
结构
- 抽象策略(Strategy)类:定义了一个公共接口,各种不同的算法以不同的方式实现这个接口,环境角色使用这个接口调用不同的算法,一般使用接口或抽象类实现。
- 具体策略(Concrete Strategy)类:实现了抽象策略定义的接口,提供具体的算法实现。
- 环境(Context)类:持有一个策略类的引用,最终给客户端调用。
实现
实现场景:登录某平台,此平台有多种登录模式
package com.wly.DesignPatterns;
/**
* @program: StudyDome
* @author: yuanzhang
* @create: 2021-01-22 10:12
**/
public class StrategyPattern {
public static void main(String[] args) {
LoginContext loginContext = new LoginContext();
System.out.println("***********使用微信登录");
loginContext.setLoginStrategy(new WeChatLogin());
loginContext.executeLogin();
System.out.println("***********使用支付宝登录");
loginContext.setLoginStrategy(new AlipayLogin());
loginContext.executeLogin();
System.out.println("***********使用邮箱登录");
loginContext.setLoginStrategy(new MailboxLogin());
loginContext.executeLogin();
}
}
/**
* @Annotation:抽象策略---登录策略
* @Author: yuanzhang
* @Date: 10:24
*/
abstract class LoginStrategy{
public abstract void login();
}
/**
* @Annotation:具体策略---微信登陆策略
* @Author: yuanzhang
* @Date: 10:25
*/
class WeChatLogin extends LoginStrategy{
@Override
public void login() {
System.out.println("实行微信登录策略");
}
}
/**
* @Annotation:具体策略---支付宝登陆策略
* @Author: yuanzhang
* @Date: 10:25
*/
class AlipayLogin extends LoginStrategy{
@Override
public void login() {
System.out.println("实行支付宝登陆策略");
}
}
/**
* @Annotation:具体策略---邮箱登录策略
* @Author: yuanzhang
* @Date: 10:26
*/
class MailboxLogin extends LoginStrategy{
@Override
public void login() {
System.out.println("实行邮箱登陆策略");
}
}
/**
* @Annotation:环境类---登录环境
* @Author: yuanzhang
* @Date: 10:26
*/
class LoginContext{
private LoginStrategy loginStrategy;
public LoginStrategy getLoginStrategy() {
return loginStrategy;
}
public void setLoginStrategy(LoginStrategy loginStrategy) {
this.loginStrategy = loginStrategy;
}
public void executeLogin(){
this.loginStrategy.login();
}
}
输出
***********使用微信登录 实行微信登录策略 ***********使用支付宝登录 实行支付宝登陆策略 ***********使用邮箱登录 实行邮箱登陆策略