0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

為什么我不再推薦枚舉策略模式?

jf_ro2CN3Fa ? 來(lái)源:芋道源碼 ? 2023-04-14 10:52 ? 次閱讀


一、為什么講策略模式

策略模式,應(yīng)該是工作中比較常用的設(shè)計(jì)模式,調(diào)用方自己選擇用哪一種策略完成對(duì)數(shù)據(jù)的操作,也就是“一個(gè)類的行為或其算法可以在運(yùn)行時(shí)更改”

我個(gè)人的理解是 將一些除了過(guò)程不同其他都一樣的函數(shù)封裝成策略,然后調(diào)用方自己去選擇想讓數(shù)據(jù)執(zhí)行什么過(guò)程策略。常見(jiàn)的例子為根據(jù)用戶分類推薦不同的排行榜(用戶關(guān)注點(diǎn)不一樣,推薦榜單就不一樣)

和單例模式一樣,隨著時(shí)間發(fā)展,我不再推薦經(jīng)典策略模式,更推薦簡(jiǎn)單策略用枚舉策略模式,復(fù)雜地用工廠策略模式。下面引入一個(gè)例子,我們的需求是:對(duì)一份股票數(shù)據(jù)列表,給出低價(jià)榜、高價(jià)榜、漲幅榜。這其中只有排序條件的區(qū)別,比較適合作為策略模式的例子

基于 Spring Boot + MyBatis Plus + Vue & Element 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項(xiàng)目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 視頻教程:https://doc.iocoder.cn/video/

二、經(jīng)典策略模式

數(shù)據(jù)DTO

@Data
publicclassStock{

//股票交易代碼
privateStringcode;

//現(xiàn)價(jià)
privateDoubleprice;

//漲幅
privateDoublerise;
}

抽象得到的策略接口

publicinterfaceStrategy{

/**
*將股票列表排序
*
*@paramsource源數(shù)據(jù)
*@return排序后的榜單
*/
Listsort(Listsource);
}

實(shí)現(xiàn)我們的策略類

/**
*高價(jià)榜
*/
publicclassHighPriceRankimplementsStrategy{

@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getPrice).reversed())
.collect(Collectors.toList());
}
}

/**
*低價(jià)榜
*/
publicclassLowPriceRankimplementsStrategy{

@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getPrice))
.collect(Collectors.toList());
}
}

/**
*高漲幅榜
*/
publicclassHighRiseRankimplementsStrategy{

@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getRise).reversed())
.collect(Collectors.toList());
}
}

經(jīng)典的Context類,

publicclassContext{
privateStrategystrategy;

publicvoidsetStrategy(Strategystrategy){
this.strategy=strategy;
}

publicListgetRank(Listsource){
returnstrategy.sort(source);
}
}

于是 我們順禮成章地得到調(diào)用類--榜單實(shí)例RankServiceImpl

@Service
publicclassRankServiceImpl{

/**
*dataService.getSource()提供原始的股票數(shù)據(jù)
*/
@Resource
privateDataServicedataService;

/**
*前端傳入榜單類型,返回排序完的榜單
*
*@paramrankType榜單類型
*@return榜單數(shù)據(jù)
*/
publicListgetRank(StringrankType){
//創(chuàng)建上下文
Contextcontext=newContext();
//這里選擇策略
switch(rankType){
case"HighPrice":
context.setStrategy(newHighPriceRank());
break;
case"LowPrice":
context.setStrategy(newLowPriceRank());
break;
case"HighRise":
context.setStrategy(newHighRiseRank());
break;
default:
thrownewIllegalArgumentException("rankTypenotfound");
}
//然后執(zhí)行策略
returncontext.getRank(dataService.getSource());
}
}

我們可以看到經(jīng)典方法,創(chuàng)建了一個(gè)接口、三個(gè)策略類,還是比較啰嗦的。調(diào)用類的實(shí)現(xiàn)也待商榷,新增一個(gè)策略類還要修改榜單實(shí)例(可以用抽象工廠解決,但是復(fù)雜度又上升了)。加之我們有更好的選擇,所以此處不再推薦經(jīng)典策略模式

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項(xiàng)目地址:https://github.com/YunaiV/yudao-cloud
  • 視頻教程:https://doc.iocoder.cn/video/

三、基于枚舉的策略模式

這里對(duì)這種簡(jiǎn)單的策略,推薦用枚舉進(jìn)行優(yōu)化。枚舉的本質(zhì)是創(chuàng)建了一些靜態(tài)類的集合。

我下面直接給出例子,大家可以直觀感受一下

枚舉策略類

publicenumRankEnum{
//以下三個(gè)為策略實(shí)例
HighPrice{
@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getPrice).reversed())
.collect(Collectors.toList());
}
},
LowPrice{
@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getPrice))
.collect(Collectors.toList());
}
},
HighRise{
@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getRise).reversed())
.collect(Collectors.toList());
}
};

//這里定義了策略接口
publicabstractListsort(Listsource);
}

對(duì)應(yīng)的調(diào)用類也得以優(yōu)化,榜單實(shí)例RankServiceImpl

@Service
publicclassRankServiceImpl{

/**
*dataService.getSource()提供原始的股票數(shù)據(jù)
*/
@Resource
privateDataServicedataService;

/**
*前端傳入榜單類型,返回排序完的榜單
*
*@paramrankType榜單類型形似RankEnum.HighPrice.name()
*@return榜單數(shù)據(jù)
*/
publicListgetRank(StringrankType){
//獲取策略,這里如果未匹配會(huì)拋IllegalArgumentException異常
RankEnumrank=RankEnum.valueOf(rankType);
//然后執(zhí)行策略
returnrank.sort(dataService.getSource());
}
}

可以看到,如果策略簡(jiǎn)單的話,基于枚舉的策略模式優(yōu)雅許多,調(diào)用方也做到了0修改,但正確地使用枚舉策略模式需要額外考慮以下幾點(diǎn)。

  • 枚舉的策略類是公用且靜態(tài),這意味著這個(gè)策略過(guò)程不能引入非靜態(tài)的部分,擴(kuò)展性受限
  • 策略模式的目標(biāo)之一,是優(yōu)秀的擴(kuò)展性和可維護(hù)性,最好能新增或修改某一策略類時(shí),對(duì)其他類是無(wú)改動(dòng)的。而枚舉策略如果過(guò)多或者過(guò)程復(fù)雜,維護(hù)是比較困難的,可維護(hù)性受限

四、基于工廠的策略模式

為了解決良好的擴(kuò)展性和可維護(hù)性,我更推薦以下利用spring自帶beanFactory的優(yōu)勢(shì),實(shí)現(xiàn)一個(gè)基于工廠的策略模式。

策略類改動(dòng)只是添加了@Service注解,并指定了Service的value屬性

/**
*高價(jià)榜
*注意申明Service.value=HighPrice,他是我們的key,下同
*/
@Service("HighPrice")
publicclassHighPriceRankimplementsStrategy{

@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getPrice).reversed())
.collect(Collectors.toList());
}
}

/**
*低價(jià)榜
*/
@Service("LowPrice")
publicclassLowPriceRankimplementsStrategy{

@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getPrice))
.collect(Collectors.toList());
}
}

/**
*高漲幅榜
*/
@Service("HighRise")
publicclassHighRiseRankimplementsStrategy{

@Override
publicListsort(Listsource){
returnsource.stream()
.sorted(Comparator.comparing(Stock::getRise).reversed())
.collect(Collectors.toList());
}
}

調(diào)用類修改較大,接入借助spring工廠特性,完成策略類

@Service
publicclassRankServiceImpl{

/**
*dataService.getSource()提供原始的股票數(shù)據(jù)
*/
@Resource
privateDataServicedataService;
/**
*利用注解@Resource@Autowired特性,直接獲取所有策略類
*key=@Service的value
*/
@Resource
privateMaprankMap;

/**
*前端傳入榜單類型,返回排序完的榜單
*
*@paramrankType榜單類型和Service注解的value屬性一致
*@return榜單數(shù)據(jù)
*/
publicListgetRank(StringrankType){
//判斷策略是否存在
if(!rankMap.containsKey(rankType)){
thrownewIllegalArgumentException("rankTypenotfound");
}
//獲得策略實(shí)例
Strategyrank=rankMap.get(rankType);
//執(zhí)行策略
returnrank.sort(dataService.getSource());
}
}

若讀者使用的不是Spring,也可以找找對(duì)應(yīng)框架的工廠模式實(shí)現(xiàn),或者自己實(shí)現(xiàn)一個(gè)抽象工廠。

工廠策略模式會(huì)比枚舉策略模式啰嗦,但也更加靈活、易擴(kuò)展性和易維護(hù)。故簡(jiǎn)單策略推薦枚舉策略模式,復(fù)雜策略才推薦工廠策略模式。



審核編輯 :李倩



聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 框架
    +關(guān)注

    關(guān)注

    0

    文章

    403

    瀏覽量

    17517
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4344

    瀏覽量

    62839

原文標(biāo)題:為什么我不再推薦枚舉策略模式?

文章出處:【微信號(hào):芋道源碼,微信公眾號(hào):芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    設(shè)計(jì)模式-策略模式

    作者:京東工業(yè) 孫磊 一、概念 策略模式(Strategy Pattern)也稱為(Policy Parttern)。 它定義了算法家族,分別封裝起來(lái),讓它們之間可以互相替換,此模式讓算法的變換
    的頭像 發(fā)表于 01-08 13:47 ?79次閱讀
    設(shè)計(jì)<b class='flag-5'>模式</b>-<b class='flag-5'>策略</b><b class='flag-5'>模式</b>

    Java 枚舉策略模式、函數(shù)式接口的結(jié)合:實(shí)現(xiàn)高內(nèi)聚低耦合的設(shè)計(jì)

    中,通常會(huì)使用枚舉來(lái)定義業(yè)務(wù)上的一組常量,那除了簡(jiǎn)單地定義常量之外,我們?nèi)绾卫?b class='flag-5'>枚舉來(lái)實(shí)現(xiàn)高內(nèi)聚、低耦合的設(shè)計(jì)呢?下面介紹下枚舉策略模式、
    的頭像 發(fā)表于 11-21 14:06 ?257次閱讀

    請(qǐng)問(wèn)什么情況下會(huì)損壞TLV2548,或者導(dǎo)致INT信號(hào)不再反應(yīng)?

    如題,請(qǐng)問(wèn)什么情況下會(huì)損壞TLV2548,或者導(dǎo)致INT信號(hào)不再反應(yīng)? 曾多次驅(qū)動(dòng)過(guò)該AD,都可以正常讀取碼值。但有兩次遇到過(guò)AD沒(méi)有任何反應(yīng),只在上電的瞬間讀取電壓,INT不再拉低(
    發(fā)表于 11-14 06:39

    蘋果調(diào)整策略:逐步摒棄年更產(chǎn)品發(fā)布模式

    10月8日訊,彭博社知名記者馬克·古爾曼在《Power On》通訊中揭示,蘋果公司正逐步轉(zhuǎn)變其傳統(tǒng)的“年度更新”發(fā)布模式,邁向更為靈活的產(chǎn)品發(fā)布策略
    的頭像 發(fā)表于 10-08 16:46 ?826次閱讀

    stm32f407 USB外接HUB怎么枚舉HUB和其他USB設(shè)備?

    stm32f407 的USB接口 想外接一個(gè)HUB 來(lái)支持更多的USB設(shè)備,但是這個(gè)HUB 該怎么枚舉,還有就是HUB下的USB設(shè)備怎么枚舉,有做過(guò)的大神嗎。求指導(dǎo),求demo。小弟拜謝了
    發(fā)表于 04-29 08:13

    STM32F103 USB枚舉不成功的原因?

    (NVIC_VectTab_FLASH,0x8800),但此時(shí)USB便枚舉不成功,此時(shí)其他中斷是正常的;但若把APP起始地址改為0X8000000不使用IAP進(jìn)行跳轉(zhuǎn)則枚舉沒(méi)有問(wèn)題.比較懷疑中斷向量這塊的設(shè)置是不是還有問(wèn)題,跪
    發(fā)表于 04-29 06:29

    stm32f103 usb枚舉問(wèn)題求解

    移植了一個(gè)USB HID設(shè)備,發(fā)現(xiàn)必須初始化usart1,這樣usb hid設(shè)備才能枚舉,否則枚舉失敗,沒(méi)有發(fā)現(xiàn)硬件上有聯(lián)系???
    發(fā)表于 04-26 07:57

    調(diào)試USB hots的時(shí)候,枚舉過(guò)不去,為什么?

    調(diào)試USB hots的時(shí)候,枚舉過(guò)不去,第一步的狀態(tài)都不對(duì)。看數(shù)據(jù)是中斷函數(shù)USBH_OTG_ISR_Handler返回的,請(qǐng)問(wèn)哪位有相關(guān)說(shuō)明發(fā)一下,現(xiàn)在總是觸發(fā)gintsts.b.sofintr中斷。謝謝!
    發(fā)表于 04-10 07:28

    CYUS3014 RAM燒寫成功,為什么枚舉不起來(lái)?

    CYUSB3014 RAM顯示燒寫成功后,缺不再枚舉,技術(shù)支持讓更換芯片后,問(wèn)題依舊,請(qǐng)問(wèn)是否還有別的方式可以查找問(wèn)題?
    發(fā)表于 02-29 07:13

    cyusb3014枚舉的工作原理是什么?如果下載固件后不枚舉,可能會(huì)是什么原因,原理是什么?

    用3014設(shè)計(jì)自己的板子,參考開(kāi)發(fā)板kit3的原理圖,插入計(jì)算機(jī)后可以被識(shí)別為Bootloader,采用USB啟動(dòng)模式,下載實(shí)例中的固件后,顯示燒寫成功,但設(shè)備在計(jì)算機(jī)中消失了,不再重新被
    發(fā)表于 02-29 07:11

    通常情況下CYUSB3014枚舉后的結(jié)果都會(huì)顯示為Bootloader,如何改變枚舉結(jié)果?

    通常情況下CYUSB3014枚舉后的結(jié)果都會(huì)顯示為Bootloader 如果向讓USB3.0連接以后,將我連接的設(shè)備識(shí)別成ChinaBeijing .請(qǐng)問(wèn)我應(yīng)該如何操作??
    發(fā)表于 02-28 08:01

    如何同時(shí)枚舉SlaveFIFO和UART(CDC)?

    使用Re: Slave FIFO + UART Driver Setup中的程序,將img下載進(jìn)FX3中,成功枚舉出了“USB串行設(shè)備(COM13)”,但是并沒(méi)有出現(xiàn)
    發(fā)表于 02-28 07:23

    如何在SlaveFifoSync例程中添加HID設(shè)備枚舉?

    您好,使用了SlaveFifoSync的32bit模式例程,現(xiàn)在想在其中添加一個(gè)HID的設(shè)備(鼠標(biāo)),程序見(jiàn)附件,參照了Can EZ-USB FX3 enumerate seri
    發(fā)表于 02-28 06:31

    cx3初始化GPIO會(huì)停止UVC枚舉的原因?

    需要使用 GPIO 為的圖像傳感器和其他設(shè)備進(jìn)行重置控制。 為此,正在嘗試在的 CSI 生成的代碼中添加簡(jiǎn)單的 GPIO 控制,但是當(dāng)我添加以下幾行時(shí),
    發(fā)表于 02-27 07:38

    cyusb3014進(jìn)入DP替代模式后斷開(kāi)USB連接是什么原因?

    使用USB3.0 + dp2lane mode),請(qǐng)問(wèn)可能是什么原因嗎? 有兩個(gè)猜測(cè): 1. 檢查 usb3.0+dp2lane 后是否會(huì)再次枚舉USB模式已協(xié)商。 枚舉失敗 2
    發(fā)表于 02-23 08:17