Java中常用的設(shè)計模式之策略模式詳解
優(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)方式
假設(shè)一個場景,我們在電商系統(tǒng)中,訂單分為很多種,例如:普通訂單,秒殺訂單,拼團訂單等等。我們需要創(chuàng)建一個訂單的時候,由于訂單的類型不同,我們需要根據(jù)訂單的類型執(zhí)行不同的業(yè)務(wù)邏輯。
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ù)訂單類型,找到對應(yīng)的處理器
*/
Optional<OrderService> any = orderServices.stream().filter(f -> f.type().getType() == type).findAny();
/**
* 沒有對應(yīng)的處理器
*/
if (!any.isPresent()) {
throw new RuntimeException("沒有找到相應(yīng)的訂單實現(xiàn)");
}
// 創(chuàng)建訂單
any.get().createOrder();
}
}
這里就能體現(xiàn)出 @Autowired 的強大之處,可以一次性自動注入多個對象。根據(jù)訂單類型,選出對應(yīng)的處理器來處理該訂單。
二、測試
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());
}
}
輸出結(jié)果

總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
詳解獲取Spring MVC中所有RequestMapping以及對應(yīng)方法和參數(shù)
本篇文章主要介紹了詳解獲取Spring MVC中所有RequestMapping以及對應(yīng)方法和參數(shù),具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-03-03
java如何實現(xiàn)自動生成數(shù)據(jù)庫設(shè)計文檔
以前我們還需要手寫數(shù)據(jù)庫設(shè)計文檔、現(xiàn)在可以通過引入screw核心包來實現(xiàn)Java?數(shù)據(jù)庫文檔一鍵生成。本文將具體介紹一下如何通過java自動生成數(shù)據(jù)庫設(shè)計文檔,需要的朋友可以參考下2021-11-11
Java基礎(chǔ)之JDBC的數(shù)據(jù)庫連接與基本操作
這篇文章主要介紹了Java基礎(chǔ)之JDBC的數(shù)據(jù)庫連接與基本操作,文中有非常詳細的代碼示例,對正在學習java基礎(chǔ)的小伙伴們也有很好的幫助,需要的朋友可以參考下2021-05-05
關(guān)于mybatis3中@SelectProvider的使用問題
這篇文章主要介紹了mybatis3中@SelectProvider的使用技巧,@SelectProvide指定一個Class及其方法,并且通過調(diào)用Class上的這個方法來獲得sql語句,本文通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下2021-12-12

