本文屬于系列文章《設(shè)計模式》,附上文集鏈接
策略模式
- 定義:定義一系列的算法,把每一個算法封裝起來, 并且使它們可相互替換嗜傅。
- 作用:首先是封裝的算法瘦锹,然后可相互替換,可以想象出一個場景方库,就是有很多種的選擇结序,然后可以選擇最合適的一種,如果不用策略模式的話纵潦,那就是一個一個自行選擇徐鹤,對的垃环。
- 屬于行為類模式
舉個例子
之前做外包做一個網(wǎng)站,其中有一個模塊是支付的返敬,可供選擇的方式有支付寶支付遂庄,微信支付和支付寶的跳轉(zhuǎn)支付(H5跳轉(zhuǎn)app支付)。當(dāng)時是蠢的啊劲赠,沒想到策略模式這種東西涛目,上代碼
// 支付控制器
public class PayController {
// 微信要自己生成二維碼,若是在手機(jī)端,則必須要微信瀏覽器才能使用
public void wechatPay() {
// 模擬request收集到訂單號,商品描述,總價錢的參數(shù)
String orderNo = System.currentTimeMillis() + "";
String body = "終極商店—大紅蘋果";
long totalFee = 6L;
// 模擬調(diào)用支付api的過程
System.out.println(
"使用訂單號:" + orderNo + "凛澎,商品描述:" + body + "和總價錢:" + totalFee + "調(diào)用微信支付工具類霹肝,請求下單API,返回支付URL并根據(jù)URL生成二維碼");
}
// 支付寶直接跳轉(zhuǎn)到支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付,只能用于PC端
public void aliPay() {
// 模擬request收集到訂單號,商品描述,總價錢的參數(shù)
String orderNo = System.currentTimeMillis() + "";
String body = "終極商店—大紅蘋果";
long totalFee = 6L;
// 模擬調(diào)用支付api的過程
System.out.println("將訂單號:" + orderNo + "塑煎,商品描述:" + body + "和總價錢:" + totalFee + "作為參數(shù)沫换,訪問支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付");
}
// 移動端使用支付寶支付,跳轉(zhuǎn)到支付寶收銀臺最铁,支付寶收銀臺有喚醒支付寶APP的函數(shù)讯赏,但不能在微信瀏覽器打開
public void mobileAliPay() {
// 模擬request收集到訂單號,商品描述,總價錢的參數(shù)
String orderNo = System.currentTimeMillis() + "";
String body = "終極商店—大紅蘋果";
long totalFee = 6L;
// 模擬調(diào)用支付api的過程
System.out.println("將訂單號:" + orderNo + ",商品描述:" + body + "和總價錢:" + totalFee + "作為參數(shù)冷尉,訪問支付寶的收銀臺來發(fā)起移動支付");
}
}
// 場景類
public class Client {
public static void main(String[] args) throws InterruptedException {
PayController payController = new PayController();
System.out.println("用戶Tom選擇了商品漱挎,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在PC端,使用Chrome瀏覽器操作");
payController.aliPay();
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Sivan選擇了商品网严,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在移動端端识樱,使用非微信瀏覽器操作");
payController.mobileAliPay();
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Jack選擇了商品,然后開始下單支付");
System.out.println("用戶選擇了微信支付");
System.out.println("在移動端端震束,使用微信瀏覽器操作");
payController.wechatPay();
}
}
結(jié)果:
用戶Tom選擇了商品怜庸,然后開始下單支付
用戶選擇了支付寶支付
在PC端,使用Chrome瀏覽器操作
將訂單號:1491293476471垢村,商品描述:終極商店—大紅蘋果和總價錢:6作為參數(shù)割疾,訪問支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付
\-----------------------------------------
用戶Sivan選擇了商品,然后開始下單支付
用戶選擇了支付寶支付
在移動端端嘉栓,使用非微信瀏覽器操作
將訂單號:1491293476482宏榕,商品描述:終極商店—大紅蘋果和總價錢:6作為參數(shù),訪問支付寶的收銀臺來發(fā)起移動支付
\-----------------------------------------
用戶Jack選擇了商品侵佃,然后開始下單支付
用戶選擇了微信支付
在移動端端麻昼,使用微信瀏覽器操作
使用訂單號:1491293476492,商品描述:終極商店—大紅蘋果和總價錢:6調(diào)用微信支付工具類馋辈,請求下單API,返回支付URL并根據(jù)URL生成二維碼
代碼就不解釋了抚芦,注釋都說明了,當(dāng)時就是傻了,沒想到擴(kuò)展性等的問題叉抡。試想下尔崔,以后如果多一種支付方式,我就要在PayController多加一個方法褥民,修改已有代碼季春,不符合開閉原則喔。其次消返,上面的每個pay的方法都有相同的代碼(requset獲取參數(shù))载弄。。一點復(fù)用都沒有撵颊,賊氣(這里倒和策略模式無關(guān)侦锯,只是對自身實力的一種吐槽)。
用策略模式改造下秦驯,如下:
// 支付策略接口
public interface PayStrategy {
public void pay(String orderNo,String body,long totalFee);
}
//支付寶支付策略,直接跳轉(zhuǎn)到支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付挣棕,只能用于PC端
public class AliPayStraegy implements PayStrategy {
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("將"
\+ "訂單號:" + orderNo + "译隘,商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "作為參數(shù),訪問支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付");
}
}
// 移動端使用支付寶支付策略洛心,跳轉(zhuǎn)到支付寶收銀臺固耘,支付寶收銀臺有喚醒支付寶APP的函數(shù),但不能在微信瀏覽器打開
public class MobileAlipayStrategy implements PayStrategy{
@Override
public void pay(String orderNo,String body,long totalFee){
// 模擬調(diào)用支付api的過程
System.out.println("將"
\+ "訂單號:" + orderNo + "词身,商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "作為參數(shù)厅目,訪問支付寶的收銀臺來發(fā)起移動支付");
}
}
//微信支付策略,要自己生成二維碼,若是在手機(jī)端法严,則必須要微信瀏覽器才能使用
public class WechatPayStrategy implements PayStrategy {
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("使用"
\+ "訂單號:" + orderNo + "损敷,商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "調(diào)用微信支付工具類,請求下單API,返回支付URL并根據(jù)URL生成二維碼");
}
}
// 支付控制器
public class PayController {
private PayStrategy payStrategy;
public void setPayStrategy(PayStrategy payStrategy) {
this.payStrategy = payStrategy;
}
public void pay() {
// 模擬request收集到訂單號,商品描述,總價錢的參數(shù)
String orderNo = System.currentTimeMillis() + "";
String body = "終極商店—大紅蘋果";
long totalFee = 6L;
// 模擬調(diào)用支付api的過程
payStrategy.pay(orderNo, body, totalFee);
}
}
// 場景類
public class Client {
public static void main(String[] args) throws InterruptedException {
PayController payController = new PayController();
System.out.println("用戶Tom選擇了商品深啤,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在PC端拗馒,使用Chrome瀏覽器操作");
payController.setPayStrategy(new AliPayStraegy());
payController.pay();
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Sivan選擇了商品,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在移動端端溯街,使用非微信瀏覽器操作");
payController.setPayStrategy(new MobileAlipayStrategy());
payController.pay();
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Jack選擇了商品诱桂,然后開始下單支付");
System.out.println("用戶選擇了微信支付");
System.out.println("在移動端端,使用微信瀏覽器操作");
payController.setPayStrategy(new WechatPayStrategy());
payController.pay();
}
}
我們先定義了一個PayStrategy的接口呈昔,然后每一個支付的具體方法都實現(xiàn)這個接口挥等,然后在PayController中組合了一個PayStrategy,定義了一個set方法堤尾,這個set方法肝劲,我把它稱作“策略選擇器”,高大上,哈哈涡相。
然后在pay方法中哲泊,調(diào)用策略來執(zhí)行pay方法就行了。然后在客戶端催蝗,每次調(diào)用支付的接口的時候切威,就使用我們的“策略選擇器”,使用指定的策略丙号,然后就OK了先朦。PayController在執(zhí)行pay方法的時候會根據(jù)傳入的PayStrategy們來選擇相應(yīng)的方法來執(zhí)行,這就是簡單的策略模式犬缨。
有啥好處喳魏?第一,以后每多一種支付方式怀薛,我只需要實現(xiàn)PayStrategy接口來新建一個類刺彩,而不需要修改原有代碼,符合開閉原則枝恋。第二创倔,假設(shè)原有的支付方式發(fā)生改變,需要修改焚碌,我只需要修改對應(yīng)的策略類畦攘,避免了對其他的策略類造成影響的可能。第三十电,客戶端對Controller的了解變少了知押,因為只需要了解策略的種類,而不需要了解Controller哪個方法具體是干啥的鹃骂,也符合迪米特原則台盯。
延伸下,當(dāng)時我在外包中是用@Resource將PayStrategy的實現(xiàn)類都放到IOC容器畏线,然后在PayController的payService(對的爷恳,并不是像代碼那樣在Controller直接組裝的,啊哈哈)象踊,用@AutoWired組裝每一個支付策略的温亲,頁面會傳一個payType參數(shù)(對的,上面的代碼還是沒提到杯矩,啊哈哈)栈虚,根據(jù)payType參數(shù)來使用相應(yīng)的策略,來完成下單那個動作史隆。如果有更好的辦法的朋友歡迎拍磚魂务,在此先謝謝了。
延伸
這個真的是延伸了,首先是看書看到的實現(xiàn)方法粘姜,覺得挺有意思的一種實現(xiàn)鬓照,叫策略枚舉,也是666.
代碼:
// 策略枚舉
public enum Pay {
//支付寶支付策略孤紧,直接跳轉(zhuǎn)到支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付豺裆,只能用于PC端
AliPay() {
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("將"
\+ "訂單號:" + orderNo + ",商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "作為參數(shù)号显,訪問支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付");
}
},
//微信支付策略臭猜,要自己生成二維碼,若是在手機(jī)端,則必須要微信瀏覽器才能使用
WechatPay() {
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("使用"
\+ "訂單號:" + orderNo + "押蚤,商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "調(diào)用微信支付工具類蔑歌,請求下單API,返回支付URL并根據(jù)URL生成二維碼");
}
},
// 移動端使用支付寶支付策略,跳轉(zhuǎn)到支付寶收銀臺揽碘,支付寶收銀臺有喚醒支付寶APP的函數(shù)次屠,但不能在微信瀏覽器打開
MobileAliPay() {
@Override
public void pay(String orderNo,String body,long totalFee){
// 模擬調(diào)用支付api的過程
System.out.println("將"
\+ "訂單號:" + orderNo + ",商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "作為參數(shù)雳刺,訪問支付寶的收銀臺來發(fā)起移動支付");
}
};
// 定義支付的抽象方法,枚舉類型每多一個值帅矗,都得實現(xiàn)這個抽象方法,就是這個特性才能666
public abstract void pay(String orderNo, String body, long totalFee);
}
// 場景類
public class Client {
public static void main(String[] args) throws InterruptedException {
// 偷偷懶煞烫,直接模擬那些參數(shù)了,勿噴
String orderNo = System.currentTimeMillis() + "";
String body = "終極商店—大紅蘋果";
long totalFee = 6L;
PayController payController = new PayController();
System.out.println("用戶Tom選擇了商品累颂,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在PC端滞详,使用Chrome瀏覽器操作");
// 直接使用策略的枚舉值
Pay.AliPay.pay(orderNo, body, totalFee);
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Sivan選擇了商品,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在移動端端紊馏,使用非微信瀏覽器操作");
// 直接使用策略的枚舉值
Pay.MobileAliPay.pay(orderNo, body, totalFee);
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Jack選擇了商品料饥,然后開始下單支付");
System.out.println("用戶選擇了微信支付");
System.out.println("在移動端端,使用微信瀏覽器操作");
// 直接使用策略的枚舉值
Pay.WechatPay.pay(orderNo, body, totalFee);
}
}
結(jié)果:
沒錯,上面就是策略枚舉赫编,當(dāng)時第一次看到的時候驚了個大呆巡蘸,枚舉還可以這樣玩。我在上面的示例偷了偷懶擂送,正確的做法應(yīng)該是在payController那里選擇策略的悦荒,具體的代碼腦補(bǔ)下吧。嘹吨。搬味。
雖然好像和很多看到的策略模式有很大的出入,但不得不說,這個并沒有什么毛病碰纬,同一樣?xùn)|西的不同表達(dá)萍聊。看回定義悦析,把每一個算法封裝起來, 并且使它們可相互替換寿桨,而他們的算法實現(xiàn)都是在枚舉值中實現(xiàn)的,相互替換也沒什么毛病她按。畢竟擴(kuò)展接口的方法也需要記住哪個類對應(yīng)哪個算法牛隅,而枚舉需要的記住某個枚舉值對應(yīng)某個算法,只不過就是如果需要構(gòu)造函數(shù)做點什么事的話酌泰,枚舉的方法就很蛋疼了媒佣。。懂的自然懂陵刹,哈哈默伍。
上面那個是看書看到,下面這個就是我看Thinking In Java中看到內(nèi)部類的時候突發(fā)奇想聯(lián)想到的了衰琐,可能看到內(nèi)部類應(yīng)該已經(jīng)想到大概怎么實現(xiàn)了吧也糊,哈哈,來看代碼:
// 使用內(nèi)部類的策略模式
public class PayStrategyWithInnerClass {
//支付寶支付策略羡宙,直接跳轉(zhuǎn)到支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付狸剃,只能用于PC端
public class AliPay implements PayStrategy{
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("將"
\+ "訂單號:" + orderNo + ",商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "作為參數(shù)狗热,訪問支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付");
}
}
//微信支付策略钞馁,要自己生成二維碼,若是在手機(jī)端,則必須要微信瀏覽器才能使用
public class WechatPay implements PayStrategy{
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("使用"
\+ "訂單號:" + orderNo + "匿刮,商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "調(diào)用微信支付工具類僧凰,請求下單API,返回支付URL并根據(jù)URL生成二維碼");
}
}
// 移動端使用支付寶支付策略,跳轉(zhuǎn)到支付寶收銀臺熟丸,支付寶收銀臺有喚醒支付寶APP的函數(shù)训措,但不能在微信瀏覽器打開
public class MobileAliPay implements PayStrategy{
@Override
public void pay(String orderNo, String body, long totalFee) {
// 模擬調(diào)用支付api的過程
System.out.println("將"
\+ "訂單號:" + orderNo + ",商品描述:" + body + "和總價錢:" + totalFee + ""
\+ "作為參數(shù)光羞,訪問支付寶的收銀臺來發(fā)起移動支付");
}
}
}
// 場景類
public class Client {
public static void main(String[] args) throws InterruptedException {
PayStrategyWithInnerClass payStrategy = new PayStrategyWithInnerClass();
PayController payController = new PayController();
System.out.println("用戶Tom選擇了商品绩鸣,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在PC端,使用Chrome瀏覽器操作");
// 實現(xiàn)方式有所差別
payController.setPayStrategy(payStrategy.new AliPay());
payController.pay();
// 直接使用策略的枚舉值
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Sivan選擇了商品纱兑,然后開始下單支付");
System.out.println("用戶選擇了支付寶支付");
System.out.println("在移動端端全闷,使用非微信瀏覽器操作");
payController.setPayStrategy(payStrategy.new MobileAliPay());
payController.pay();
Thread.sleep(10);
System.out.println("-----------------------------------------");
System.out.println("用戶Jack選擇了商品,然后開始下單支付");
System.out.println("用戶選擇了微信支付");
System.out.println("在移動端端萍启,使用微信瀏覽器操作");
payController.setPayStrategy(payStrategy.new WechatPay());
payController.pay();
}
}
結(jié)果:
用戶Tom選擇了商品总珠,然后開始下單支付
用戶選擇了支付寶支付
在PC端屏鳍,使用Chrome瀏覽器操作
將訂單號:1491315431218,商品描述:終極商店—大紅蘋果和總價錢:6作為參數(shù)局服,訪問支付寶收集訂單信息的jsp頁面來發(fā)起網(wǎng)關(guān)支付
\-----------------------------------------
用戶Sivan選擇了商品钓瞭,然后開始下單支付
用戶選擇了支付寶支付
在移動端端,使用非微信瀏覽器操作
將訂單號:1491315431230淫奔,商品描述:終極商店—大紅蘋果和總價錢:6作為參數(shù)山涡,訪問支付寶的收銀臺來發(fā)起移動支付
\-----------------------------------------
用戶Jack選擇了商品,然后開始下單支付
用戶選擇了微信支付
在移動端端唆迁,使用微信瀏覽器操作
使用訂單號:1491315431241鸭丛,商品描述:終極商店—大紅蘋果和總價錢:6調(diào)用微信支付工具類,請求下單API,返回支付URL并根據(jù)URL生成二維碼
其實也沒啥不同唐责,內(nèi)部類來實現(xiàn)策略PayStrategy接口鳞溉,然后實現(xiàn)方法,在調(diào)用的場景也是實例化一個內(nèi)部類而已鼠哥,實現(xiàn)的實質(zhì)也是策略選擇器熟菲。。內(nèi)部類那里的外圍類有點像枚舉朴恳,但是又有點差別抄罕。沒遇到實際問題也想不出來啥例子,有經(jīng)驗的人士麻煩評論區(qū)拍個磚于颖,感激不盡呆贿。
以上就是策略模式,水平有限森渐,難免有錯做入,歡迎評論區(qū)指責(zé)