Spring?boot?運(yùn)用策略模式實(shí)現(xiàn)避免多次使用if的操作代碼
前言
這里就不詳細(xì)去介紹策略模式是怎么樣的了,想了解的可以點(diǎn)擊下面的鏈接
策略模式介紹的鏈接:策略模式的介紹
這里列出策略模式的好處
場(chǎng)景:某網(wǎng)頁(yè)有個(gè)支付,其中包含了微信、支付寶等方式的支付方式 ,后續(xù)明確還會(huì)進(jìn)行兼容其他的支付方式
用策略模式的好處:
- 避免多次使用if判斷具體是用哪種支付策略進(jìn)行操作。
- 因?yàn)槊糠N策略(微信支付、支付寶支付)的內(nèi)容都比較復(fù)雜。策略模式能將每種策略分離出來(lái),方面后續(xù)維護(hù)管理
下面我們將使用Spring boot 運(yùn)用策略模式,實(shí)現(xiàn)上面的需求
環(huán)境配置
- JDK8
- Spring boot 2.3.7.RELEASE
- 整合了spring-boot-starter-web
實(shí)現(xiàn)目標(biāo)
使用策略模式后,新加一種支付策略時(shí),只需要在策略枚舉中添加新加的策略信息,外加一個(gè)策略類(lèi)即可,而不再需要添加新的if判斷。
準(zhǔn)備策略接口和具體實(shí)現(xiàn)策略類(lèi)
支付策略接口
/**
* 支付策略
*/
public interface PayStrategy {
/**
* 支付(參數(shù)就沒(méi)具體寫(xiě)了,可以定義成每個(gè)支付必須要有的參數(shù))
* @return
*/
boolean pay();
}微信支付策略類(lèi)
/**
* 第三方——微信支付(這里注意我修改了Bean的默認(rèn)命名)
*/
@Component("wechatPayStrategy")
public class WeChatPayStrategyImpl implements PayStrategy{
/**
* 支付
* @return
*/
@Override
public boolean pay() {
//進(jìn)行微信的支付邏輯
System.out.println("正在進(jìn)行微信的支付邏輯");
return true;
}
}支付寶支付策略類(lèi)
/**
* 支付寶第三方支付(這里注意我修改了Bean的默認(rèn)命名)
*/
@Component("alipayStrategy")
public class AliPayStrategyImpl implements PayStrategy {
/**
* 支付寶支付
* @return
*/
@Override
public boolean pay() {
//進(jìn)行支付寶支付邏輯
System.out.println("進(jìn)行支付寶支付邏輯");
return true;
}
}上述已將各自的策略的處理類(lèi)進(jìn)行了分離,接下來(lái)時(shí)使用支付策略工廠類(lèi)和支付策略上下文將各自的策略類(lèi)聯(lián)系起來(lái)
準(zhǔn)備支付策略上下文Context和支付策略工廠類(lèi)
支付策略工廠類(lèi)
package com.example.springboot_strategy.strategy.factory;
import com.example.springboot_strategy.enums.PayStrategyEnum;
import com.example.springboot_strategy.strategy.PayStrategy;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import javax.swing.plaf.synth.SynthTextAreaUI;
import java.util.Map;
/**
* 支付策略工廠類(lèi)
*/
@Component
public class PayStrategyFactory {
/**
* 通過(guò)Spring容器的方式注入
*/
@Resource
private Map<String, PayStrategy> payStrategyMap;
/**
* 獲取對(duì)應(yīng)支付策略類(lèi)
* @param payStrategyEnum 支付策略枚舉
*/
public PayStrategy getPayStrategy(PayStrategyEnum payStrategyEnum){
if(!payStrategyMap.containsKey(payStrategyEnum.getClassName())){
System.out.println("沒(méi)有對(duì)應(yīng)的支付策略,無(wú)法進(jìn)行支付");
return null;
}
return payStrategyMap.get(payStrategyEnum.getClassName());
}
}這里工廠類(lèi)的邏輯是利用了Spring容器的處理方式,如果有多種類(lèi)同時(shí)實(shí)現(xiàn)了某個(gè)接口,那么可以使用Map集合接收,Map對(duì)應(yīng)的泛型,String是Bean名稱(chēng),PayStrategy是每個(gè)具體實(shí)現(xiàn)類(lèi),這樣我們就可以使用Bean類(lèi)型去指定具體的策略類(lèi)了,然后建立一個(gè)支付策略枚舉去管理這些Bean名稱(chēng)。同時(shí),也可以將Bean名稱(chēng)與客戶端定義的類(lèi)型進(jìn)行關(guān)聯(lián)。
支付策略枚舉類(lèi)
/**
* 支付策略類(lèi)型
*/
public enum PayStrategyEnum {
WECHAT_PAY("wechat","wechatPayStrategy","微信支付"),
ALIPAY("alipay","alipayStrategy","支付寶支付")
;
/**
* 支付策略code
*/
private String code;
/**
* bean名稱(chēng)
*/
private String className;
/**
* 信息
*/
private String info;
PayStrategyEnum(String code,String className,String info){
this.code=code;
this.className=className;
this.info=info;
}
public String getCode() {
return code;
}
public String getClassName() {
return className;
}
public String getInfo() {
return info;
}
}上面枚舉類(lèi)中code代表的是客戶端定義的類(lèi)型(例如我從前端接收到的支付type,這個(gè)type可以是這個(gè)code),className顧名思義,指的是每種策略的bean名稱(chēng),info是代表每種策略的內(nèi)容
支付策略上下文
/**
* 支付策略上下文
*/
@Component
public class PayStrategyContext {
@Autowired
private PayStrategyFactory payStrategyFactory;
/**
* 支付執(zhí)行
* @param payDTO 支付參數(shù)
* @return
*/
public boolean payHandle(PayDTO payDTO){
//將某屬性的值轉(zhuǎn)換成具體的枚舉。這里是根據(jù)payDTO的type字段對(duì)應(yīng)枚舉的code進(jìn)行轉(zhuǎn)換
Optional<PayStrategyEnum> payStrategyEnumOptional = Arrays.stream(PayStrategyEnum.class.getEnumConstants())
.filter((e) -> e.getCode().equals(payDTO.getType())).findAny();
if(!payStrategyEnumOptional.isPresent()){
System.out.println("匹配不到具體支付策略");
return false;
}
PayStrategyEnum payStrategyEnum = payStrategyEnumOptional.get();
PayStrategy payStrategy = payStrategyFactory.getPayStrategy(payStrategyEnum);
//進(jìn)行payDto參數(shù)的處理.....
boolean pay = payStrategy.pay();
//支付后的記錄處理..
return true;
}
}pageDto類(lèi)
/**
* 支付DTO
*/
public class PayDTO {
/**
* 支付類(lèi)型
*/
private String type;
/**
* 支付金額
*/
private BigDecimal payMoney;
/**
* ...........
*/
public String getType() {
return type;
}
public void setType(String type) {
this.type = type;
}
public BigDecimal getPayMoney() {
return payMoney;
}
public void setPayMoney(BigDecimal payMoney) {
this.payMoney = payMoney;
}
}這個(gè)策略上下文,則是選擇策略的入口,這里會(huì)進(jìn)行參數(shù)的處理,將這里我就將pageDTO類(lèi)中的type字符串轉(zhuǎn)換成對(duì)應(yīng)的枚舉類(lèi)。
到這里使用策略模式的編寫(xiě)算是完成了,下面進(jìn)行編寫(xiě)客戶端的代碼
客戶端代碼
支付控制器
@RestController
@RequestMapping("pay")
public class PayController {
@Autowired
private PayStrategyContext payStrategyContext;
@PostMapping
public boolean pay(@RequestBody PayDTO payDTO){
//這里因?yàn)閼?。。就沒(méi)有加上Service層了,直接在控制器處理
return payStrategyContext.payHandle(payDTO);
}
}效果

新需求
后續(xù)新增一個(gè)銀聯(lián)的支付方式,我們只需要添加銀聯(lián)的支付策略類(lèi)和添加銀聯(lián)的支付枚舉即可實(shí)現(xiàn)
添加銀聯(lián)的支付策略類(lèi)
/**
* 銀聯(lián)支付(這里注意我修改了Bean的默認(rèn)命名)
*/
@Component("unionPayStrategy")
public class UnionPayStrategyImp implements PayStrategy {
/**
* 銀聯(lián)支付
* @return
*/
@Override
public boolean pay() {
//進(jìn)行銀聯(lián)的支付
System.out.println("進(jìn)行銀聯(lián)的支付邏輯");
return true;
}
}
復(fù)制代碼在枚舉類(lèi)中添加銀聯(lián)的支付枚舉
/**
* 支付策略類(lèi)型
*/
public enum PayStrategyEnum {
WECHAT_PAY("wechat","wechatPayStrategy","微信支付"),
ALIPAY("alipay","alipayStrategy","支付寶支付"),
UNION_PAY("unionpay","unionPayStrategy","銀聯(lián)支付")
;
/**
* 支付策略code
*/
private String code;
/**
* bean名稱(chēng)
*/
private String className;
/**
* 信息
*/
private String info;
PayStrategyEnum(String code,String className,String info){
this.code=code;
this.className=className;
this.info=info;
}
public String getCode() {
return code;
}
public String getClassName() {
return className;
}
public String getInfo() {
return info;
}
}實(shí)現(xiàn)效果

以上是我使用Spring boot 運(yùn)用策略模式實(shí)現(xiàn)的效果,如果有誤人子弟的地方,望在評(píng)論區(qū)指出。
到此這篇關(guān)于Spring boot 運(yùn)用策略模式實(shí)現(xiàn),避免多次使用if的文章就介紹到這了,更多相關(guān)Spring boot 策略模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java輸入流Scanner/BufferedReader使用方法示例
這篇文章主要介紹了Java輸入流Scanner/BufferedReader使用方法,大家看示例吧2013-11-11
Java 匿名對(duì)象與匿名內(nèi)部類(lèi)的使用
很多小伙伴對(duì)匿名對(duì)象和匿名內(nèi)部類(lèi)的寫(xiě)法有點(diǎn)陌生,本文主要介紹了Java 匿名對(duì)象與匿名內(nèi)部類(lèi)的使用,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01
Java類(lèi)庫(kù)BeanUtils組件使用方法及實(shí)例詳解
這篇文章主要介紹了Java類(lèi)庫(kù)BeanUtils組件使用方法級(jí)實(shí)例詳解,需要的朋友可以參考下2020-02-02
Java用jxl讀取excel并保存到數(shù)據(jù)庫(kù)的方法
這篇文章主要為大家詳細(xì)介紹了Java用jxl讀取excel并保存到數(shù)據(jù)庫(kù)的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10
SpringBoot參數(shù)驗(yàn)證10個(gè)技巧值得收藏
Spring Boot提供了內(nèi)置的驗(yàn)證注解,可以幫助簡(jiǎn)單、快速地對(duì)輸入字段進(jìn)行驗(yàn)證,例如檢查 null 或空字段、強(qiáng)制執(zhí)行長(zhǎng)度限制、使用正則表達(dá)式驗(yàn)證模式以及驗(yàn)證電子郵件地址,那么在Spring Boot應(yīng)用中如何做好參數(shù)校驗(yàn)工作呢,本文提供了10個(gè)小技巧感興趣的朋友一起看看吧2023-08-08
SpringMVC文件上傳原理及實(shí)現(xiàn)過(guò)程解析
這篇文章主要介紹了SpringMVC文件上傳原理及實(shí)現(xiàn)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07
深入淺析 Spring Security 緩存請(qǐng)求問(wèn)題
這篇文章主要介紹了 Spring Security 緩存請(qǐng)求問(wèn)題,本文通過(guò)實(shí)例文字相結(jié)合的形式給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2019-04-04
Java基礎(chǔ)學(xué)習(xí)之Swing事件監(jiān)聽(tīng)
今天學(xué)習(xí)java的Swing庫(kù),創(chuàng)建桌面應(yīng)用的時(shí)候,突然發(fā)現(xiàn)有些按鈕需要特定的功能響應(yīng),故來(lái)研究一番Swing的事件監(jiān)聽(tīng),文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-05-05

