設(shè)計(jì)模式之工廠模式—要的是工廠而不是作坊
工廠模式應(yīng)該是我們比較常用的設(shè)計(jì)模式之一,它提供了一種創(chuàng)建對象的最佳方式,在創(chuàng)建對象時(shí)不會(huì)對調(diào)用者暴露創(chuàng)建邏輯,調(diào)用者只需根據(jù)自己的需求獲取需要的對象,做到"拿來即用"。
說人話就是,我要什么,你就得給什么。
我們用生活中常用的支付來說明工廠模式的實(shí)現(xiàn)邏輯。我們在用手機(jī)支付的時(shí)候都會(huì)選擇支付方式,比如微信支付、支付寶支付,當(dāng)然還有其他的比如微付充支付、連連支付等等。用戶選擇了某一種支付方式,相應(yīng)的要生成支付的對象,執(zhí)行支付的方法,而這里支付對象的生成,我們交給工廠去做。
首先定義一個(gè)支付的抽象類:
抽象產(chǎn)品類
這個(gè)類的作用是標(biāo)準(zhǔn)化支付對象,規(guī)定好支付對象可以進(jìn)行的操作,那就是下單。
然后就可以具體化我們的支付對象了,這里就以微信支付和支付寶支付為例:
微信支付產(chǎn)品類
支付寶支付產(chǎn)品類
有了具體的支付產(chǎn)品類,我們就可以創(chuàng)建工廠進(jìn)行生產(chǎn)了,將微信支付和支付寶支付的支付對象給創(chuàng)建出來:
工廠類
很簡單,就是根據(jù)支付類型返回支付對象。
接下來做一個(gè)小測試,模擬下用戶支付,假設(shè)用戶選擇了微信支付:
測試方法
可以看到最終執(zhí)行了微信的下單方法。
OK,大功告成,可以下班啦!
但這真的是我們要的工廠模式嗎?
假如我們需要再加一種支付方式,比如連連支付,怎么整?簡單啊,再加個(gè)連連支付的產(chǎn)品類,工廠類里再加個(gè)if判斷,返回連連支付的對象,不就OK了嗎?
這樣是可以實(shí)現(xiàn)的。但如果我們在工廠類里進(jìn)行擴(kuò)展的時(shí)候,一不小心留個(gè)bug在里面,有可能會(huì)導(dǎo)致整個(gè)工廠類都不可用了,進(jìn)而連帶著整個(gè)支付系統(tǒng)就不可用了,這樣顯然是不合理的。實(shí)際上,這樣的工廠模式違背了一個(gè)很重要的設(shè)計(jì)原則——開閉原則。開閉原則就是說,當(dāng)你在擴(kuò)展程序的時(shí)候,不要改動(dòng)原有代碼,要實(shí)現(xiàn)一個(gè)熱插拔的效果。
這種工廠模式實(shí)際上稱為簡單工廠模式,是一個(gè)管理混亂、雜亂無章的小作坊,而我們需要的,是一個(gè)井然有序、收放自如的現(xiàn)代化工廠。
所以,這就延伸出了另一種工廠模式——工廠方法模式。
既然在原有的工廠類上進(jìn)行擴(kuò)展是不合理的,那索性就把工廠類分開,一個(gè)支付一個(gè)工廠類,各自工廠類生產(chǎn)各自的支付對象,當(dāng)需要進(jìn)行擴(kuò)展的時(shí)候,只需新建自己的工廠類和支付產(chǎn)品類,而與其他的支付方式無關(guān),這樣就實(shí)現(xiàn)了熱插拔的效果。
于是我們的代碼可以演變?yōu)?,保留原有的支付抽象類和支付產(chǎn)品類,去掉工廠類,新建一個(gè)抽象工廠類,微信支付和支付寶支付各自實(shí)現(xiàn)其工廠類。
抽象工廠類
支付寶支付工廠類
微信支付工廠類
這樣就實(shí)現(xiàn)了工廠類的分離,微信支付工廠類負(fù)責(zé)生產(chǎn)微信支付的產(chǎn)品,支付寶支付工廠類負(fù)責(zé)生產(chǎn)支付寶支付的產(chǎn)品,所有工廠類各司其職,互不干擾。
測試一下:
這樣,工廠方法模式就實(shí)現(xiàn)了,這其實(shí)也是我們平時(shí)用的最多的模式,平時(shí)所說工廠模式實(shí)際就是指的工廠方法模式。當(dāng)然,工廠模式還有抽象工廠方法模式,這是比小作坊、小工廠更加高級(jí)的超級(jí)工廠模式,實(shí)現(xiàn)起來比較繁瑣,平時(shí)也基本用不到,這里就不細(xì)說了。