設(shè)計模式 -- 策略模式+Spring Bean代替if/else
策略模式
一、什么是策略模式
- 策略模式屬于對象的行為模式。其用意是針對一組算法瓤狐,將每一個算法封裝到具有共同接口的獨立的類中乱陡,從而使得它們可以相互替換。策略模式使得算法可以在不影響到客戶端的情況下發(fā)生變化盖袭。
二、策略模式的結(jié)構(gòu)
- 策略模式是對算法的包裝彼宠,是把使用算法的責(zé)任和算法本身分割開來鳄虱,委派給不同的對象管理。策略模式通常把一個系列的算法包裝到一系列的策略類里面凭峡,作為一個抽象策略類的子類拙已。用一句話來說,就是:“準(zhǔn)備一組算法摧冀,并將每一個算法封裝起來倍踪,使得它們可以互換”。下面就以一個示意性的實現(xiàn)講解策略模式實例的結(jié)構(gòu)索昂。
這個模式涉及到三個角色:
● 環(huán)境(Context)角色:持有一個Strategy的引用建车。
● 抽象策略(Strategy)角色:這是一個抽象角色,通常由一個接口或抽象類實現(xiàn)椒惨。此角色給出所有的具體策略類所需的接口缤至。
● 具體策略(ConcreteStrategy)角色:包裝了相關(guān)的算法或行為。
使用策略模式完美的解決了我們if else 太多導(dǎo)致代碼臃腫的問題 如:
if(teachinType == 1){
……
}else if(teachinType == 2){
……
}else{
}
三康谆、實戰(zhàn)
介紹:在我們項目實戰(zhàn)中领斥,我們通常會遇到 我們后臺拿到前端傳來的參數(shù)走不同的邏輯 (線上 嫉到,線下 業(yè)務(wù) ) ,返回相同的數(shù)據(jù)結(jié)構(gòu):
if(teachinType == 1){
……
return list<Integer>;
}else if(teachinType == 2){
……
return list<Integer>;
}else{
……
return list<Integer>;
}
普通策略模式
- 定義接口
public interface Type {
/**
* 獲取類型月洛,返回集合
*/
List<Integer> getType();
}
- 定義OfflineType實現(xiàn)類實現(xiàn)Type
@Service
public class OfflineType implements Type {
@Override
public List<Integer> getType() {
List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
return list;
}
}
- 定義OnlineType實現(xiàn)類 實現(xiàn)Type接口
@Service
public class OnlineType implements Type {
@Override
public List<Integer> getType() {
List<Integer> list = new ArrayList<>();
list.add(3);
list.add(4);
return list;
}
}
- 定義策略工廠類
public class TypeFactory{
private static TypeFactory typeFactory = new TypeFactory();
//定義Map
private static Map<Integer,Type> map = new HashMap<>();
static{
map.put(1,new OfflineType());
map.put(2,new OnlineType())
}
public static TypeFactory getInstance(){
return typeFactory;
}
public Type getByType(Integer type){
return map.get(type);
}
}
- 使用
@GetMapping("list")
public List<Integer> list(@ApiParam(value ="類型")@RequestParam(required = false) Integer type) {
return TypeFactory.getInstance().getByType(type).getType();
);
接口測試輸入1何恶,輸出 1,2
接口測試輸入2嚼黔,輸出 3细层,4
策略模式+Spring Bean代替if/else
- 在上面策略模式中如果在實現(xiàn)類中使用
@Autowired
private Dao dao;
就會出現(xiàn) 空指針的情況,
java.lang.NullPointerException
因為 發(fā)現(xiàn)策略實現(xiàn)中的Dao接口注入的都是Null唬涧。
我們回頭看spring源碼 以及 他的介紹
必須在所有使用了dao的地方,包括調(diào)用它的servcie都要進(jìn)行@Autowired注入,否則之后的注入就會失敗
順著這句話疫赎,發(fā)現(xiàn)上述使用處不就沒使用@Autowired進(jìn)行注入嗎?但是這個策略模式得new熬糇洹虚缎?
我們還有其他方式:通過給實現(xiàn)類起別名撵彻,通過@Qualifier注解獲取不同的實現(xiàn)類
改造:
實現(xiàn)類
@Service("offlineType")
public class OfflineType implements Type {
@Autowired
private OfflineDao dao;
@Override
public List<Integer> getType() {
return dao.selectList();
}
}
實現(xiàn)類
@Service("onlineType")
public class OnlineType implements Type {
@Autowired
private OnlineDao dao;
@Override
public List<Integer> getType() {
return dao.selectList();
}
}
工廠
public class TypeFactory{
@Resource
@Qualifier("offlineType")
private OnlineType offlineType;
@Resource
@Qualifier("onlineType")
private OnlineType onlineType;
public Type getByType(Integer type){
Map<Integer, Type> map =new HashMap<>();
map.put(1,offlineType);
map.put(2,onlineType);
return map.get(type);
}
}
使用
// 工廠注入
@Autowired
private TypeFactory typeFactory;
@GetMapping("list")
public List<Integer> list(@ApiParam(value ="類型")@RequestParam(required = false) Integer type) {
return typeFactory.getByType(type).getType();
);
發(fā)現(xiàn) 不會出現(xiàn)空指針钓株,IOC正常注入,數(shù)據(jù)正常返回