偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

別再滿屏的 if/else 了,試試策略模式,真香!!

開發(fā) 前端
棧長(zhǎng)在開發(fā)人員的代碼中看過(guò)太多這樣的低級(jí)代碼了,真的太 low,極不好維護(hù),本文棧長(zhǎng)就教你如何用策略模式干掉 if/ else/ switch,讓你的代碼更優(yōu)雅。

 [[399910]]

你還在寫滿屏的 if/ else/ switch 之類的判斷邏輯嗎?

棧長(zhǎng)在開發(fā)人員的代碼中看過(guò)太多這樣的低級(jí)代碼了,真的太 low,極不好維護(hù),本文棧長(zhǎng)就教你如何用策略模式干掉 if/ else/ switch,讓你的代碼更優(yōu)雅。

什么是策略模式?

比如說(shuō)對(duì)象的某個(gè)行為,在不同場(chǎng)景中有不同的實(shí)現(xiàn)方式,這樣就可以將這些實(shí)現(xiàn)方式定義成一組策略,每個(gè)實(shí)現(xiàn)類對(duì)應(yīng)一個(gè)策略,在不同的場(chǎng)景就使用不同的實(shí)現(xiàn)類,并且可以自由切換策略。

策略模式結(jié)構(gòu)圖如下:

策略模式需要一個(gè)策略接口,不同的策略實(shí)現(xiàn)不同的實(shí)現(xiàn)類,在具體業(yè)務(wù)環(huán)境中僅持有該策略接口,根據(jù)不同的場(chǎng)景使用不同的實(shí)現(xiàn)類即可。

面向接口編程,而不是面向?qū)崿F(xiàn)。

策略模式的優(yōu)點(diǎn):

1、干掉繁瑣的 if、switch 判斷邏輯;

2、代碼優(yōu)雅、可復(fù)用、可讀性好;

3、符合開閉原則,擴(kuò)展性好、便于維護(hù);

策略模式的缺點(diǎn):

1、策略如果很多的話,會(huì)造成策略類膨脹;

2、使用者必須清楚所有的策略類及其用途;

策略模式實(shí)戰(zhàn)

舉個(gè)實(shí)際的例子,XX 公司是做支付的,根據(jù)不同的客戶類型會(huì)有不同的支付方式和支付產(chǎn)品,比如:信用卡、本地支付,而本地支付在中國(guó)又有微信支付、支付寶、云閃付、等更多其他第三方支付公司,這時(shí)候策略模式就派上用場(chǎng)了。

傳統(tǒng)的 if/ else/ switch 等判斷寫法大家都會(huì)寫,這里就不貼代碼了,直接看策略模式怎么搞!

1、定義策略接口

定義一個(gè)策略接口,所有支付方式的接口。

策略接口:

  1. /** 
  2.  * 支付接口 
  3.  * @author: 棧長(zhǎng) 
  4.  * @from: 公眾號(hào)Java技術(shù)棧 
  5.  */ 
  6. public interface IPayment { 
  7.  
  8.     /** 
  9.      * 支付 
  10.      * @param order 
  11.      * @return 
  12.      */ 
  13.     PayResult pay(Order order); 
  14.  

訂單信息類:

  1. /** 
  2.  * 訂單信息 
  3.  * @author: 棧長(zhǎng) 
  4.  * @from: 公眾號(hào)Java技術(shù)棧 
  5.  */ 
  6. @Data 
  7. public class Order { 
  8.  
  9.     /** 
  10.      * 金額 
  11.      */ 
  12.     private int amount; 
  13.  
  14.     /** 
  15.      * 支付類型 
  16.      */ 
  17.     private String paymentType; 
  18.  

返回結(jié)果類:

  1. /** 
  2.  * @author: 棧長(zhǎng) 
  3.  * @from: 公眾號(hào)Java技術(shù)棧 
  4.  */ 
  5. @Data 
  6. @AllArgsConstructor 
  7. public class PayResult { 
  8.  
  9.     /** 
  10.      * 支付結(jié)果 
  11.      */ 
  12.     private String result; 
  13.  

2、定義各種策略

定義各種支付策略,微信支付、支付寶、云閃付等支付實(shí)現(xiàn)類都實(shí)現(xiàn)這個(gè)接口。

微信支付實(shí)現(xiàn):

  1. /** 
  2.  * 微信支付 
  3.  * @author: 棧長(zhǎng) 
  4.  * @from: 公眾號(hào)Java技術(shù)棧 
  5.  */ 
  6. @Service("WechatPay"
  7. public class WechatPay implements IPayment { 
  8.  
  9.     @Override 
  10.     public PayResult pay(Order order) { 
  11.         return new PayResult("微信支付成功"); 
  12.     } 
  13.  

支付寶實(shí)現(xiàn):

  1. /** 
  2.  * 支付寶 
  3.  * @author: 棧長(zhǎng) 
  4.  * @from: 公眾號(hào)Java技術(shù)棧 
  5.  */ 
  6. @Service("Alipay"
  7. public class Alipay implements IPayment { 
  8.  
  9.     @Override 
  10.     public PayResult pay(Order order) { 
  11.         return new PayResult("支付寶支付成功"); 
  12.     } 
  13.  

云閃付實(shí)現(xiàn):

  1. /** 
  2.  * 銀聯(lián)云閃付 
  3.  * @author: 棧長(zhǎng) 
  4.  * @from: 公眾號(hào)Java技術(shù)棧 
  5.  */ 
  6. @Service("UnionPay"
  7. public class UnionPay implements IPayment { 
  8.  
  9.     @Override 
  10.     public PayResult pay(Order order) { 
  11.         return new PayResult("云閃付支付成功"); 
  12.     } 
  13.  

這里我把所有支付方式類都用 @Service 注解生成 Bean 放入 Spring Bean 容器中了,在使用策略的時(shí)候就不用 new 支付對(duì)象了,可以直接使用 Bean,這樣更貼近業(yè)務(wù)。Spring 基礎(chǔ)教程就不介紹了,大家可以關(guān)注公眾號(hào)Java技術(shù)棧,回復(fù):spring,歷史教程我都整理好了。

3、使用策略

有的文章使用了枚舉、HashMap 的方式來(lái)根據(jù)策略名稱映射策略實(shí)現(xiàn)類 ,這樣是沒有問題,但在使用了 Spring 框架的項(xiàng)目還是有點(diǎn)多此一舉,完全可以發(fā)揮 Spring 框架的優(yōu)勢(shì),使用 Bean 名稱就能找到對(duì)應(yīng)的策略實(shí)現(xiàn)類了。

參考示例代碼如下:

  1. /** 
  2.  * 支付服務(wù) 
  3.  * @author: 棧長(zhǎng) 
  4.  * @from: 公眾號(hào)Java技術(shù)棧 
  5.  */ 
  6. @RestController 
  7. public class PayService { 
  8.  
  9.     @Autowired 
  10.     private ApplicationContext applicationContext; 
  11.  
  12.     /** 
  13.      * 支付接口 
  14.      * @param amount 
  15.      * @param paymentType 
  16.      * @return 
  17.      */ 
  18.     @RequestMapping("/pay"
  19.     public PayResult pay(@RequestParam("amount"int amount, 
  20.                     @RequestParam("paymentType") String paymentType) { 
  21.         Order order = new Order(); 
  22.         order.setAmount(amount); 
  23.         order.setPaymentType(paymentType); 
  24.  
  25.         // 根據(jù)支付類型獲取對(duì)應(yīng)的策略 bean 
  26.         IPayment payment = applicationContext.getBean(order.getPaymentType(), IPayment.class); 
  27.  
  28.         // 開始支付 
  29.         PayResult payResult = payment.pay(order); 
  30.  
  31.         return payResult; 
  32.     } 
  33.  

看示例代碼,我并沒有像策略模式結(jié)構(gòu)圖中那樣新建一個(gè) Context 類持有策略接口,那是標(biāo)準(zhǔn)的策略模式,其實(shí)道理是一樣的,關(guān)鍵是怎么施放策略。

測(cè)試一下:

http://localhost:8080/pay?amount=8800&paymentType=WechatPay

測(cè)試 OK,傳入不同的支付方式會(huì)調(diào)用不同的策略。

本節(jié)教程所有實(shí)戰(zhàn)源碼已上傳到這個(gè)倉(cāng)庫(kù):https://github.com/javastacks/javastack

策略模式在 JDK 中的應(yīng)用

現(xiàn)在我們知道如何使用策略模式了,現(xiàn)在我們?cè)倏纯?JDK 哪些地方運(yùn)用了策略模式呢。

1、線程池中的拒絕策略

線程池的構(gòu)造中有一個(gè)拒絕策略參數(shù),默認(rèn)是默認(rèn)拒絕策略:

其實(shí)這就是一個(gè)策略接口:

下面有幾種拒絕策略的實(shí)現(xiàn):

image-20210329161322406

在創(chuàng)建線程池的時(shí)候,就可以傳入不同的拒絕策略,這就是 JDK 中策略模式的經(jīng)典實(shí)現(xiàn)了。

2、比較器

JDK 中大量使用了 Comparator 這個(gè)策略接口:

策略接口有了,但策略需要開發(fā)人員自己定。

集合排序我們比較熟悉的了,不同的排序規(guī)則其實(shí)就是不同的策略:

這個(gè)策略模式使用了函數(shù)式編程接口,比較規(guī)則使用匿名內(nèi)部類或者 Lambda 表達(dá)式就搞定了,不需要每個(gè)規(guī)則定義一個(gè)實(shí)現(xiàn)類,這樣就大量省略策略類了。

這個(gè)策略模式可能藏的比較深,但也是 JDK 中經(jīng)典的策略模式的應(yīng)用了。

不限于這兩個(gè),其實(shí)還有更多,你還知道別的么?歡迎留言分享……

所以說(shuō),策略模式就在你身邊,你一直都在用,但可能沒有發(fā)覺。。

總結(jié)

使用策略模式,我們可以輕松干掉大量的 if/ else,代碼也更優(yōu)雅,還能很靈活的擴(kuò)展。

像本文中支付的案例,后面我們想添加、刪除多少個(gè)支付方式都不用修改現(xiàn)有的代碼,所以就不會(huì)影響現(xiàn)有的業(yè)務(wù),真正做到對(duì)擴(kuò)展開放,對(duì)修改關(guān)閉。

當(dāng)然,完全干掉 if/ else 是不可能的,不能過(guò)度設(shè)計(jì),不能為了使用設(shè)計(jì)模式而使用設(shè)計(jì)模式,否則適得其反。但是,我們每個(gè)程序員都需要掌握策略模式,做到在系統(tǒng)中靈活駕馭,這樣才能寫出更優(yōu)雅、高質(zhì)量的代碼。

本節(jié)教程所有實(shí)戰(zhàn)源碼已上傳到這個(gè)倉(cāng)庫(kù):

https://github.com/javastacks/javastack

本文轉(zhuǎn)載自微信公眾號(hào)「Java技術(shù)?!梗梢酝ㄟ^(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系Java技術(shù)棧公眾號(hào)。

 

責(zé)任編輯:武曉燕 來(lái)源: Java技術(shù)棧
相關(guān)推薦

2020-06-15 08:12:51

try catch代碼處理器

2024-08-07 10:34:46

2025-02-17 10:30:01

2023-03-28 08:58:47

分庫(kù)分表TiDB

2021-04-13 06:39:13

代碼重構(gòu)code

2021-03-10 07:20:43

if-else靜態(tài)代碼

2022-07-11 08:16:55

策略模式if-else

2020-06-04 09:18:52

CTOif-else代碼

2025-05-15 03:00:00

2022-12-27 08:01:09

設(shè)計(jì)模式https://mp

2020-04-09 08:29:50

編程語(yǔ)言事件驅(qū)動(dòng)

2022-03-08 13:46:22

httpClientHTTP前端

2025-05-07 00:00:00

CSS單位JavaScript

2021-04-27 20:04:11

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

2025-07-04 01:55:00

CSS樣式Chrome

2022-05-27 21:56:55

索引存儲(chǔ)MySQL 存儲(chǔ)引擎

2022-06-14 10:49:33

代碼優(yōu)化Java

2022-01-13 10:45:59

if-else代碼Java

2024-03-11 08:21:49

2020-12-04 10:05:00

Pythonprint代碼
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)