Golang設(shè)計(jì)模式——簡(jiǎn)單工廠模式
背景
假設(shè)我們?cè)谧鲆豢钚⌒头g軟件,軟件可以將德語(yǔ)胆数、英語(yǔ)、日語(yǔ)都翻譯成目標(biāo)中文互墓,并顯示在前端必尼。
思路
我們會(huì)有三個(gè)具體的語(yǔ)言翻譯結(jié)構(gòu)體,或許以后還有更多篡撵,但現(xiàn)在分別是GermanTranslater判莉、EnglishTranslater、JapaneseTranslater育谬,他們都共同實(shí)現(xiàn)了一個(gè)接口Translator券盅。
//翻譯接口
type Translator interface {
Translate(string) string
}
//德語(yǔ)翻譯類
type GermanTranslator struct{}
func (*GermanTranslator) Translate(words string) string {
return "德語(yǔ)"
}
//英語(yǔ)翻譯類
type EnglishTranslator struct{}
func (*EnglishTranslator) Translate(words string) string {
return "英語(yǔ)"
}
//日語(yǔ)翻譯類
type JapaneseTranslator struct{}
func (*JapaneseTranslator) Translate(words string) string {
return "日語(yǔ)"
}
接下來在程序入口獲取用戶輸入的文本,并將其翻譯
package main
import (
"fmt"
"time"
)
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Println(err)
}
time.Sleep(3 * time.Second)
}()
var lan int
fmt.Printf("%s\r\n%s\r\n", "以下是可翻譯的語(yǔ)言種類膛檀,請(qǐng)輸入代表數(shù)字", "1:德語(yǔ)锰镀、2:英語(yǔ)、3:日語(yǔ)")
fmt.Scanln(&lan)
fmt.Println("請(qǐng)輸入要翻譯成中文的文本:")
var inputWords string
fmt.Scanln(&inputWords)
var translator Translator
//根據(jù)不同的語(yǔ)言種類咖刃,實(shí)例化不同的翻譯類
switch lan {
case 1:
translator = new(GermanTranslator)
case 2:
translator = new(EnglishTranslator)
case 3:
translator = new(JapaneseTranslator)
default:
panic("no such translator")
}
fmt.Println(translator.Translate(inputWords))
}
運(yùn)行結(jié)果
缺點(diǎn)
違背了開閉原則泳炉,以后還可能有法語(yǔ)、俄語(yǔ)嚎杨、阿拉伯語(yǔ)等其他翻譯器胡桃,每一次添加翻譯器都要在客戶端代碼增加對(duì)應(yīng)的switch分支,維護(hù)成本高磕潮。倘若還有不止一處調(diào)用了創(chuàng)建邏輯翠胰,還要維護(hù)多處代碼。
違背了單一職責(zé)原則自脯,客戶端處理類的職責(zé)應(yīng)該只是負(fù)責(zé)接收用戶的輸入并將其打印之景,現(xiàn)在還負(fù)責(zé)翻譯類的創(chuàng)建邏輯,導(dǎo)致這個(gè)類的職責(zé)過多膏潮。
改善
為了解決每次新增翻譯類都要修改客戶端的問題锻狗,我們引入一個(gè)很重要的設(shè)計(jì)原則,可以說每種設(shè)計(jì)模式都遵循著這個(gè)設(shè)計(jì)原則焕参。
設(shè)計(jì)原則:找出用中可能需要變化之處轻纪,并把它們獨(dú)立出來,不要和那些不需要變化的代碼混在一起叠纷。
這樣的話刻帚,每次當(dāng)新的需求來臨,我們只會(huì)改動(dòng)到那些需要變化的地方涩嚣,而不變的地方就不會(huì)被改動(dòng)影響到崇众。
顯然,翻譯應(yīng)用中容易變化的地方是生成翻譯類的邏輯航厚,因此我們把這部分職責(zé)抽出來顷歌,把它交給另外一個(gè)類去做(一般是一個(gè)靜態(tài)方法),這個(gè)類就叫翻譯工廠幔睬。而客戶端再需要生成翻譯類實(shí)例時(shí)眯漩,僅需調(diào)用翻譯工廠提供的方法即可。就算以后翻譯工廠會(huì)提供更多的翻譯類麻顶,也不會(huì)修改到客戶端的代碼赦抖,因此也就有了我們的現(xiàn)在的簡(jiǎn)單工廠模式。
簡(jiǎn)單工廠模式(Simple Factory Pattern)
又稱為靜態(tài)工廠方法(Static Factory Method)模式澈蚌,它屬于類創(chuàng)建型模式摹芙。在簡(jiǎn)單工廠模式中,可以根據(jù)參數(shù)的不同返回不同類的實(shí)例宛瞄。簡(jiǎn)單工廠模式專門定義一個(gè)類來負(fù)責(zé)創(chuàng)建其他類的實(shí)例浮禾,被創(chuàng)建的實(shí)例通常都具有共同的父類。
UML類圖
于是我們根據(jù)簡(jiǎn)單工廠模式再完善之前的代碼份汗,如下所示盈电。
工廠代碼
func Create(lan int) Translator {
var translator Translator
//根據(jù)不同的語(yǔ)言種類,實(shí)例化不同的翻譯類
switch lan {
case 1:
translator = new(GermanTranslator)
case 2:
translator = new(EnglishTranslator)
case 3:
translator = new(JapaneseTranslator)
default:
panic("no such translator")
}
return translator
}
客戶端代碼
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Println(err)
}
time.Sleep(3 * time.Second)
}()
var lan int
fmt.Printf("%s\r\n%s\r\n", "以下是可翻譯的語(yǔ)言種類杯活,請(qǐng)輸入代表數(shù)字", "1:德語(yǔ)匆帚、2:英語(yǔ)、3:日語(yǔ)")
fmt.Scanln(&lan)
fmt.Println("請(qǐng)輸入要翻譯成中文的文本:")
var inputWords string
fmt.Scanln(&inputWords)
//客戶端只關(guān)注如何獲取翻譯類旁钧,而不用關(guān)注創(chuàng)建翻譯類的細(xì)節(jié)
translator:=CreateTranslator(lan)
fmt.Println(translator.Translate(inputWords))
}
優(yōu)點(diǎn)
將客戶端和創(chuàng)建產(chǎn)品實(shí)例解耦開來吸重,使客戶端只需要關(guān)注如何獲取實(shí)例互拾。
符合單一職責(zé)。
缺點(diǎn)
- 增加新翻譯類時(shí)還是需要改動(dòng)工廠類嚎幸,沒有符合開閉原則颜矿。
應(yīng)用場(chǎng)景
當(dāng)在代碼里看到switch的時(shí)候,就應(yīng)該思考是否用簡(jiǎn)單工廠模式嫉晶。
作者:胡金生
版權(quán)所有骑疆,歡迎保留原文鏈接進(jìn)行轉(zhuǎn)載:)