设计模式之策略模式

什么是策略模式?

策略模式主要的目的是将算法的定义和调用分开,将算法定义在不同的策略类中,算法的调用根据具体的条件来动态调用策略类。

策略模式的模型图

client(算法调用者): 维护一个策略类的引用,根据不同的条线来选择调用具体的策略实现 Strategy(策略抽象类): 是所有策略类的父类,可以是抽象类亦可以是接口,抽象策略类定义的方法需要具体的策略实现类去实现。 StrategyA/StrategyA(策略具体实现类): 实现策略接口或者继承策略抽象类,实现具体的算法,可以根据具体的业务实现动态的增减策略实现类。

策略模式应用示例

需求:系统需要对接不同的银行来实现融资申请的操作,每个银行都有相应的数据加密算法对请求参数进行加密并调用相关SDK进行通信,客户端通过界面选择找哪家银行进行融资申请,根据银行的简称找到具体的银行实现类,进行融资申请的推送。

代码示例:

java 复制代码
// 定义策略父类(可以是接口也可是抽象类)
public interface FinancingHandler {
    void sendFinancingApply(FinancingRequest financingRequest);
}

// 策略实现类
@Component(value = "cmbc")
@Slf4j
public class CMBCHandler implements FinancingHandler{
    @Override
    public void sendFinancingApply(FinancingRequest financingRequest) {
        log.info("向工商银行推送融资申请信息....");
        log.info("融资请求参数:{}",financingRequest);
    }
}

// 策略实现类
@Component(value = "cbc")
@Slf4j
public class CBCHandler implements FinancingHandler{
    @Override
    public void sendFinancingApply(FinancingRequest financingRequest) {
        log.info("向建设银行推送融资申请信息....");
        log.info("融资请求参数:{}",financingRequest);
    }
}
// 枚举常量
public enum BankEnum {
    CMBC("cmbc", "中国工商银行"),
    CBC("cbc", "中国建设银行");

    private String simpleName;
    private String bankName;

    BankEnum(String simpleName, String bankName) {
        this.simpleName = simpleName;
        this.bankName = bankName;
    }
    public String getSimpleName() {
        return simpleName;
    }
}
// 融资请求实体
@Data
@ToString
public class FinancingRequest {
    private String contractNo;
    private BigDecimal amount;
    private String applyName;
    private String applyAccount;
}
// 定义策略实现类的获取方法
@Component
public class SpringApplicationContextUtil implements ApplicationContextAware {

    private static ApplicationContext context;
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        context = applicationContext;
    }


    public  FinancingHandler getFinancingHandler(String beanName) {
        return (FinancingHandler) context.getBean(beanName);
    }
}
// 业务调用入口
@Component
public class FinancingBusService {

    @Autowired
    private SpringApplicationContextUtil springApplicationContextUtil;

    public void sendFinancingApply(FinancingRequest financingRequest, BankEnum bankEnum){
        FinancingHandler financingHandler = springApplicationContextUtil.getFinancingHandler(bankEnum.getSimpleName());
        financingHandler.sendFinancingApply(financingRequest);
    }

}

// 单元测试
@SpringBootTest
public class FinancingTest {

    @Autowired
    private FinancingBusService financingBusService;

    @Test
    public void test1(){
        FinancingRequest request = new FinancingRequest();
        request.setApplyName("tony");
        request.setApplyAccount("111111");
        request.setAmount(new BigDecimal(10000));
        financingBusService.sendFinancingApply(request, BankEnum.CMBC);
    }

}

预期展示

适用场景以及优缺点

适用场景

一种需求或者一种场景,需要动态的选择多种算法中的某一种算法,这些算法都是抽象父类的一种实现。

优缺点

优点

  1. 策略模式符合开闭原则,可通过实现父类的方法来动态的增减新的子类。
  2. 策略模式可以将重复的公共代码统一移动到父类中,子类只负责实现差异的算法,能更好的实现代码的复用。

缺点

  1. 调用者必须清楚的知道不同算法之间的差异,并自行选择调用那种策略。
  2. 不支持同时使用多个策略类。
相关推荐
无心水17 小时前
深入Java线程池:BlockingQueue实现全景解析与实战指南
java·后端·面试
Java水解17 小时前
Rust 性能优化实战:从 unsafe 使用到 SIMD 指令,让服务端响应快 2 倍
后端·rust
Java水解17 小时前
JAVA面试题大全(200+道题目)
java·后端·面试
编程岁月17 小时前
java面试-0305-java线程调度方法?sleep()和wait()区别?
java·开发语言·面试
卷福同学17 小时前
AI浏览器comet拉新,一单20美元(附详细教程)
人工智能·后端
大鱼七成饱18 小时前
掌握 anyhow,让你的 Rust 错误处理优雅又安全
后端·rust
2301_7720935618 小时前
高并发webserver_interview
运维·服务器·数据库·后端·网络协议·mysql·wireshark
HashTang18 小时前
不用再配服务器了!这套 Next.js + Cloudflare 模板,一个人搞定全栈出海
前端·后端·边缘计算
水淹萌龙20 小时前
玩转 Go 表达式引擎:expr 实战指南
开发语言·后端·golang
Yeats_Liao21 小时前
Go Web 编程快速入门 07.4 - 模板(4):组合模板与逻辑控制
开发语言·后端·golang