Java中常用的設計模式之策略模式詳解
優(yōu)點
1.算法可以自由切換。
2.避免使用多重條件判斷。
3.擴展性良好。
缺點
1.策略類會增多。
2.所有策略類都需要對外暴露。
使用場景
1.如果在一個系統(tǒng)里面有許多類,它們之間的區(qū)別僅在于它們的行為,那么使用策略模式可以動態(tài)地讓一個對象在許多行為中選擇一種行為。
2.一個系統(tǒng)需要動態(tài)地在幾種算法中選擇一種。
3.如果一個對象有很多的行為,如果不用恰當?shù)哪J剑@些行為就只好使用多重的條件選擇語句來實現(xiàn)。
一、實現(xiàn)方式
假設一個場景,我們在電商系統(tǒng)中,訂單分為很多種,例如:普通訂單,秒殺訂單,拼團訂單等等。我們需要創(chuàng)建一個訂單的時候,由于訂單的類型不同,我們需要根據(jù)訂單的類型執(zhí)行不同的業(yè)務邏輯。
1、訂單類型枚舉類
package com.asurplus.common.strategy; import lombok.AllArgsConstructor; import lombok.Getter; /** * 訂單類型枚舉類 */ @Getter @AllArgsConstructor public enum OrderTypeEnum { COMMON(1001, "普通訂單"), SECKILL(1002, "秒殺訂單"), SPELL(1003, "拼團訂單"); int type; String desc; }
我們的訂單分為三種,普通訂單,秒殺訂單,拼團訂單。
2、訂單處理接口
package com.asurplus.common.strategy; /** * 訂單處理接口 */ public interface OrderService { /** * 創(chuàng)建訂單 * * @return */ void createOrder(); /** * 獲取訂單類型 * * @return */ OrderTypeEnum type(); }
3、普通訂單處理器
package com.asurplus.common.strategy; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Service; /** * 普通訂單處理器 */ @Slf4j @Service public class CommonOrderServiceImpl implements OrderService { @Override public void createOrder() { log.info("創(chuàng)建 普通訂單"); } @Override public OrderTypeEnum type() { return OrderTypeEnum.COMMON; } }
4、秒殺訂單處理器
package com.asurplus.common.strategy; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Service; /** * 秒殺訂單處理器 */ @Slf4j @Service public class SeckillOrderServiceImpl implements OrderService { @Override public void createOrder() { log.info("創(chuàng)建 秒殺訂單"); } @Override public OrderTypeEnum type() { return OrderTypeEnum.SECKILL; } }
5、拼團訂單處理器
package com.asurplus.common.strategy; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Service; /** * 拼團訂單處理器 */ @Slf4j @Service public class SpellOrderServiceImpl implements OrderService { @Override public void createOrder() { log.info("創(chuàng)建 拼團訂單"); } @Override public OrderTypeEnum type() { return OrderTypeEnum.SPELL; } }
6、下單管理器
package com.asurplus.common.strategy; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import java.util.List; import java.util.Optional; /** * 訂單管理器 */ @Component public class OrderManager { /** * Autowired 注解的強大之處 */ @Autowired private List<OrderService> orderServices; /** * 創(chuàng)建訂單 * * @param type 訂單類型 * @return */ public void createOrder(int type) { /** * 根據(jù)訂單類型,找到對應的處理器 */ Optional<OrderService> any = orderServices.stream().filter(f -> f.type().getType() == type).findAny(); /** * 沒有對應的處理器 */ if (!any.isPresent()) { throw new RuntimeException("沒有找到相應的訂單實現(xiàn)"); } // 創(chuàng)建訂單 any.get().createOrder(); } }
這里就能體現(xiàn)出 @Autowired 的強大之處,可以一次性自動注入多個對象。根據(jù)訂單類型,選出對應的處理器來處理該訂單。
二、測試
1、引入依賴
<!-- 測試依賴 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> </dependency>
2、測試用例
package com.asurplus.common.strategy; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; /** * 策略模式 */ @RunWith(SpringRunner.class) @SpringBootTest public class TestMain { @Autowired private OrderManager orderManager; @Test public void test() { // 創(chuàng)建 秒殺訂單 orderManager.createOrder(OrderTypeEnum.SECKILL.getType()); } }
輸出結果
總結
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關注腳本之家的更多內容!
相關文章
詳解獲取Spring MVC中所有RequestMapping以及對應方法和參數(shù)
本篇文章主要介紹了詳解獲取Spring MVC中所有RequestMapping以及對應方法和參數(shù),具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-03-03java如何實現(xiàn)自動生成數(shù)據(jù)庫設計文檔
以前我們還需要手寫數(shù)據(jù)庫設計文檔、現(xiàn)在可以通過引入screw核心包來實現(xiàn)Java?數(shù)據(jù)庫文檔一鍵生成。本文將具體介紹一下如何通過java自動生成數(shù)據(jù)庫設計文檔,需要的朋友可以參考下2021-11-11Java基礎之JDBC的數(shù)據(jù)庫連接與基本操作
這篇文章主要介紹了Java基礎之JDBC的數(shù)據(jù)庫連接與基本操作,文中有非常詳細的代碼示例,對正在學習java基礎的小伙伴們也有很好的幫助,需要的朋友可以參考下2021-05-05關于mybatis3中@SelectProvider的使用問題
這篇文章主要介紹了mybatis3中@SelectProvider的使用技巧,@SelectProvide指定一個Class及其方法,并且通過調用Class上的這個方法來獲得sql語句,本文通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下2021-12-12