成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

知道策略模式!但不會在項目里使用?

開發 前端
簡單的來說當做某個事情有多個方式的時候,可以抽象為接口,然后每個實現是一種解決方式,由調用方來選擇不同的實現方式。

前言

在學開發的第二年就開始聽說要想代碼寫得好,一定要會設計模式。于是就興致沖沖的啃了《Head First 設計模式》,看完之后對于策略模式映像很深刻,覺得這個模式好,易上手,應用廣,我又能優化一波代碼了,于是興致沖沖的打開了我的 IDEA,開整!!!

策略模式初體驗(錯誤示范)

在講訴我的策略模式首秀前,我們先回顧下策略模式的基本概念。

策略模式

  • 意圖:定義一系列的算法,把它們一個個封裝起來, 并且使它們可相互替換。
  • 主要解決:在有多種算法相似的情況下,使用 if…else 所帶來的復雜和難以維護。
  • 何時使用:一個系統有許多許多類,而區分它們的只是他們直接的行為。

簡單的來說當做某個事情有多個方式的時候,可以抽象為接口,然后每個實現是一種解決方式,由調用方來選擇不同的實現方式。

理解了后我開始對我們的代碼進行了重構,當時我第一家公司有這樣一段代碼,大概是這個意思(時間長了,我憑記憶重寫的)。

有這樣一個抽獎的方法,我們后臺控制中獎率,不同的時候我們會調整不同的中獎策略。

public class NumStrategy {

enum RandomEnum{
/**
* 平均策略
*/
AVERAGE,
/**
* 80%的幾率中獎
*/
RANDOM28;
}
/**
* 抽獎方法,根據不同的策略進行抽獎
* @param randomEnum
* @return true:代表中獎 false:代表沒中獎
*/
public boolean luckDraw(RandomEnum randomEnum){
if(randomEnum.equals(RandomEnum.AVERAGE)){
Random random = new Random();
int num = random.nextInt(100);
return num >= 50;
}else if(randomEnum.equals(RandomEnum.RANDOM28)){
Random random = new Random();
int num = random.nextInt(100);
return num >= 20;
}
return false;
}
}

我一看,這不就是妥妥的策略模式嗎?開搞。

一頓改造之后變成了這樣:

public class NumStrategy2 {
enum RandomEnum{
/**
* 平均策略
*/
AVERAGE,
/**
* 80%的幾率中獎
*/
RANDOM28;
}
/**
* 抽獎方法,根據不同的策略進行抽獎
* @param randomEnum
* @return
public boolean luckDraw(RandomEnum randomEnum){
if(randomEnum.equals(RandomEnum.AVERAGE)){
return new AverageStrategy().luckDraw();
}else if(randomEnum.equals(RandomEnum.RANDOM28)){
return new Random28Strategy().luckDraw();
}
return false;
}
interface LuckDrawStrategy{
boolean luckDraw();
}
class AverageStrategy implements LuckDrawStrategy{
@Override
public boolean luckDraw(){
Random random = new Random();
int num = random.nextInt(100);
return num >= 50;
}
}
class Random28Strategy implements LuckDrawStrategy{
@Override
public boolean luckDraw(){
Random random = new Random();
int num = random.nextInt(100);
return num >= 20;
}
}

}

改造完成之后我滿意的提交了代碼,但是在組長 review 的時候給我又改了回來。說你整這么多類干嘛?我理直氣壯的說我這是用策略模式優化代碼。他說沒必要,先改回去吧。

我憤憤的接受了,但心里想著:哎,你連策略模式都不懂?

經過這么多年,我開始理解我當時的做法其實不對,本來很簡單的代碼,而且里面的邏輯不會有變動,其實不需要抽象出來。我的改動有過度設計之嫌。把原來的30行代碼搞成了80行

一報還一報,這幾年我見過太多次當年的我這樣寫代碼的了。

即為了用設計模式而用設計模式。而忘了設計模式的初衷是為了代碼更易理解,更可靠,更易維護。

甚至還見過有人學了策略模式后說要把項目里所有的 if/else都安排上策略模式。

梅開二度

又過了一年多,在一次面試的時候,也有著關于策略模式的討論。

【面試官】問:你說你用過策略模式,請問你為什么用它?

【我】:為了抽離各個不同實現邏輯,優化 if/else,使代碼更簡單易懂

【面試官】:你具體說說,怎么去掉的 if/else

【我】:內心 OS(背的知識點,我也好久沒用了啊)。我硬著頭皮說,我可以使用工廠模式+策略模式來做。

【面試官】:那你工廠模式的那里不是也要用 if/else 判斷嗎?

【我】:。。。額。唔。。。那確實還是要用到 if/else

把我問住了,我支支吾吾的回答確實還是要 if/else 來判斷一次,只不過把判斷移到了工廠模式里面去了。

我下來后又去實踐了下,想著放在 map 里行不行呢?

public class NumStrategy3 {
enum RandomEnum{
/**
* 平均策略
*/
AVERAGE,
/**
* 80%的幾率中獎
*/
RANDOM28;
}
static Map<RandomEnum,LuckDrawStrategy> map = new HashMap<>();
static{
map.put(RandomEnum.RANDOM28,new Random28Strategy());
map.put(RandomEnum.AVERAGE,new AverageStrategy());
}
/**
* 抽獎方法,根據不同的策略進行抽獎
* @param randomEnum
* @return
public boolean luckDraw(RandomEnum randomEnum){
LuckDrawStrategy luckDrawStrategy = map.get(randomEnum);
return luckDrawStrategy.luckDraw();
}
interface LuckDrawStrategy{
boolean luckDraw();
}
static class AverageStrategy implements LuckDrawStrategy{
@Override
public boolean luckDraw(){
Random random = new Random();
int num = random.nextInt(100);
return num >= 50;
}
}
static class Random28Strategy implements LuckDrawStrategy{
@Override
public boolean luckDraw(){
Random random = new Random();
int num = random.nextInt(100);
return num >= 20;
}
}
}

終于是解決了 if/else 的情況,不過這樣很短的 if/else,里面邏輯不怎么變動時,我個人是不建議用策略模式,這里只是示例。

推薦用法

又過了幾年,當初的菜鳥也成長為了一個老鳥。

當時項目里有這樣一個代碼:

下面的代碼我進行了一些簡化,我們有一個功能,對頁面上的指標進行計算,不同的指標對應不同的計算方法。頁面上指標一期做 4 個,后續會做到十幾個。

public interface TransferService {
String transfer();
}
@Service
public class SearchTransformService {
@Autowired
private UserTransferService userTransferService;
@Autowired
private AgeTransferService ageTransferService;
@Autowired
private InterestTransferService interestTransferService;
/**
* 根據不同的編碼進行轉換
* @param code
* @return
public String transform(String code){
if(code.equals("user")){
return userTransferService.transfer();
}else if(code.equals("age")){
return ageTransferService.transfer();
}else if(code.equals("interest")){
return interestTransferService.transfer();
}
return "";
}
}

可以看到這樣的業務場景下,這樣的寫法 if/else 就會很長,后續十幾個的情況下就很難維護。另外 code 使用的是魔數,也是不好的一種寫法。我對此進行了優化如下:

先將 code 用枚舉定義

   enum CodeEnum {
USER("user"),
AGE("age"),
INTEREST("interest"),
;
private String code;

public String getCode() {
return code;
}
CodeEnum(String code) {
this.code = code;
}
private static final Map<String, CodeEnum> map = Arrays.stream(CodeEnum.values()).collect(Collectors.toMap(CodeEnum::getCode, Function.identity()));
public CodeEnum of(String code) {
return map.get(code);
}
}

原有的接口上增加一個transCode方法,每個實現需要聲明是對應哪個編碼的實現

public interface TransferService {

String transfer();

CodeEnum transCode();
}

@Service
public class AgeTransferService implements TransferService {
@Override
public String transfer(){
return null;
}

@Override
public CodeEnum transCode(){
return CodeEnum.AGE;
}
}

使用 map 存儲編碼對應的實現類的關聯關系,以此來獲取對應的轉換器實現類

@Service
public class SearchTransformService implements InitializingBean {
@Autowired
private List<TransferService> transferServiceList;
private Map<CodeEnum, TransferService> transferServiceMap;
@Override
// 項目啟動時將實現類放入到map中去
public void afterPropertiesSet() throws Exception {
transferServiceMap = transferServiceList.stream().collect(Collectors.toMap(TransferService::transCode, Function.identity()));
}
/**
* 根據不同的編碼進行轉換
* @param code
* @return
public String transform(String code){
TransferService transferService = transferServiceMap.get(CodeEnum.of(code));
Assert.notNull(transferService,"找不到對應的轉換器");
return transferService.transfer();
}
}

重構后是不是就很簡潔了呢?如果后續新增新的編碼轉換器,只需要先在枚舉里定義,然后新增實現類實現方法就行了,不需要關心是怎么調用的,只關心具體的實現邏輯,降低了維護成本。

這才是策略模式的真正應用吧。不要再亂用了,哈哈哈。

責任編輯:姜華 來源: 今日頭條
相關推薦

2025-04-07 13:17:49

2018-07-16 09:36:05

2022-06-23 11:22:12

LinuxLinusLinux 社區

2012-06-20 14:38:32

2010-08-26 10:06:18

微軟Hyper-VVMworld

2009-03-13 19:04:57

Windows 7發布日期

2018-04-27 15:00:54

2022-05-05 19:00:22

PythonPython特性

2010-10-19 10:44:49

李開復

2019-10-15 16:02:23

戴爾

2020-06-28 14:18:51

Android 11功能谷歌

2020-07-20 13:59:17

服務器項目工程打包

2015-07-17 11:35:22

Windows 10功能

2011-11-03 16:09:36

2013-12-27 15:42:34

小米閃購

2021-10-10 13:44:09

微信支付iPhone手機移動應用

2020-05-13 08:45:19

數據中心IT云服務

2020-05-18 09:43:53

數據中心云服務云計算

2020-06-28 14:51:15

容器Gartner軟件提供商

2022-07-05 08:05:00

策略模式接口實現類
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 一区二区三区日韩 | 精品国产欧美一区二区三区成人 | 999观看免费高清www | 三区四区在线观看 | 久久精品中文字幕 | 国产综合久久 | 9191成人精品久久 | 欧美一级在线观看 | 国产黄色小视频在线观看 | 成人在线不卡 | 国产成人一区二区三区 | 中文字幕1区2区3区 亚洲国产成人精品女人久久久 | 久久精品国产一区二区三区 | 日本网站免费在线观看 | 国产日韩av一区二区 | 人人干在线视频 | 日本一区二区三区精品视频 | 亚洲一av | 精品国产伦一区二区三区观看体验 | 日韩成人精品 | 神马影院一区二区三区 | 狠狠久 | 久久另类 | 久久综合亚洲 | 久久精品亚洲 | 91精品91久久久 | 日本欧美国产在线观看 | 视频在线一区二区 | 亚洲精品视频免费 | 一区二区三区四区在线视频 | www久久99| 亚洲欧美v| 国产精品久久久久久 | 欧美日韩国产一区二区三区 | 欧美精品一区二区三区在线播放 | 亚洲精品在线看 | 欧美日韩在线高清 | 色桃网 | 亚洲激情av | 黄色成人免费在线观看 | 免费看a|