0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

GoF給裝飾者模式的定義

元閏子的邀請(qǐng) ? 來源:元閏子的邀請(qǐng) ? 作者:元閏子的邀請(qǐng) ? 2022-06-29 10:22 ? 次閱讀

上一篇:【Go實(shí)現(xiàn)】實(shí)踐GoF的23種設(shè)計(jì)模式:原型模式

簡單的分布式應(yīng)用系統(tǒng)(示例代碼工程):https://github.com/ruanrunxue/Practice-Design-Pattern--Go-Implementation

簡介

我們經(jīng)常會(huì)遇到“給現(xiàn)有對(duì)象/模塊新增功能”的場景,比如 http router 的開發(fā)場景下,除了最基礎(chǔ)的路由功能之外,我們常常還會(huì)加上如日志、鑒權(quán)、流控等 middleware。如果你查看框架的源碼,就會(huì)發(fā)現(xiàn) middleware 功能的實(shí)現(xiàn)用的就是裝飾者模式(Decorator Pattern)。

GoF給裝飾者模式的定義如下:

Decorators provide a flexible alternative to subclassing for extending functionality. Attach additional responsibilities to an object dynamically.

簡單來說,裝飾者模式通過組合的方式,提供了能夠動(dòng)態(tài)地給對(duì)象/模塊擴(kuò)展新功能的能力。理論上,只要沒有限制,它可以一直把功能疊加下去,具有很高的靈活性。

如果寫過 Java,那么一定對(duì) I/O Stream 體系不陌生,它是裝飾者模式的經(jīng)典用法,客戶端程序可以動(dòng)態(tài)地為原始的輸入輸出流添加功能,比如按字符串輸入輸出,加入緩沖等,使得整個(gè) I/O Stream 體系具有很高的可擴(kuò)展性和靈活性。

UML 結(jié)構(gòu)

0c30719e-f700-11ec-ba43-dac502259ad0.jpg

場景上下文

在簡單的分布式應(yīng)用系統(tǒng)(示例代碼工程)中,我們?cè)O(shè)計(jì)了 Sidecar 邊車模塊,它的用處主要是為了 1)方便擴(kuò)展network.Socket的功能,如增加日志、流控等非業(yè)務(wù)功能;2)讓這些附加功能對(duì)業(yè)務(wù)程序隱藏起來,也即業(yè)務(wù)程序只須關(guān)心看到network.Socket接口即可。

0c4d81d0-f700-11ec-ba43-dac502259ad0.jpg

代碼實(shí)現(xiàn)

Sidecar 的這個(gè)功能場景,很適合使用裝飾者模式來實(shí)現(xiàn),代碼如下:

//demo/network/socket.go
packagenetwork

//關(guān)鍵點(diǎn)1:定義被裝飾的抽象接口
//Socket網(wǎng)絡(luò)通信Socket接口
typeSocketinterface{
//Listen在endpoint指向地址上起監(jiān)聽
Listen(endpointEndpoint)error
//Close關(guān)閉監(jiān)聽
Close(endpointEndpoint)
//Send發(fā)送網(wǎng)絡(luò)報(bào)文
Send(packet*Packet)error
//Receive接收網(wǎng)絡(luò)報(bào)文
Receive(packet*Packet)
//AddListener增加網(wǎng)絡(luò)報(bào)文監(jiān)聽者
AddListener(listenerSocketListener)
}

//關(guān)鍵點(diǎn)2:提供一個(gè)默認(rèn)的基礎(chǔ)實(shí)現(xiàn)
typesocketImplstruct{
listenerSocketListener
}

funcDefaultSocket()*socketImpl{
return&socketImpl{}
}

func(s*socketImpl)Listen(endpointEndpoint)error{
returnInstance().Listen(endpoint,s)
}
...//socketImpl的其他Socket實(shí)現(xiàn)方法


//demo/sidecar/flowctrl_sidecar.go
packagesidecar

//關(guān)鍵點(diǎn)3:定義裝飾器,實(shí)現(xiàn)被裝飾的接口
//FlowCtrlSidecarHTTP接收端流控功能裝飾器,自動(dòng)攔截Socket接收?qǐng)?bào)文,實(shí)現(xiàn)流控功能
typeFlowCtrlSidecarstruct{
//關(guān)鍵點(diǎn)4:裝飾器持有被裝飾的抽象接口作為成員屬性
socketnetwork.Socket
ctx*flowctrl.Context
}

//關(guān)鍵點(diǎn)5:對(duì)于需要擴(kuò)展功能的方法,新增擴(kuò)展功能
func(f*FlowCtrlSidecar)Receive(packet*network.Packet){
httpReq,ok:=packet.Payload().(*http.Request)
//如果不是HTTP請(qǐng)求,則不做流控處理
if!ok{
f.socket.Receive(packet)
return
}
//流控后返回429TooManyRequest響應(yīng)
if!f.ctx.TryAccept(){
httpResp:=http.ResponseOfId(httpReq.ReqId()).
AddStatusCode(http.StatusTooManyRequest).
AddProblemDetails("enterflowctrlstate")
f.socket.Send(network.NewPacket(packet.Dest(),packet.Src(),httpResp))
return
}
f.socket.Receive(packet)
}

//關(guān)鍵點(diǎn)6:不需要擴(kuò)展功能的方法,直接調(diào)用被裝飾接口的原生方法即可
func(f*FlowCtrlSidecar)Close(endpointnetwork.Endpoint){
f.socket.Close(endpoint)
}
...//FlowCtrlSidecar的其他方法

//關(guān)鍵點(diǎn)7:定義裝飾器的工廠方法,入?yún)楸谎b飾接口
funcNewFlowCtrlSidecar(socketnetwork.Socket)*FlowCtrlSidecar{
return&FlowCtrlSidecar{
socket:socket,
ctx:flowctrl.NewContext(),
}
}

//demo/sidecar/all_in_one_sidecar_factory.go
//關(guān)鍵點(diǎn)8:使用時(shí),通過裝飾器的工廠方法,把所有裝飾器和被裝飾者串聯(lián)起來
func(aAllInOneFactory)Create()network.Socket{
returnNewAccessLogSidecar(NewFlowCtrlSidecar(network.DefaultSocket()),a.producer)
}

總結(jié)實(shí)現(xiàn)裝飾者模式的幾個(gè)關(guān)鍵點(diǎn):

  1. 定義需要被裝飾的抽象接口,后續(xù)的裝飾器都是基于該接口進(jìn)行擴(kuò)展。
  2. 為抽象接口提供一個(gè)基礎(chǔ)實(shí)現(xiàn)。
  3. 定義裝飾器,并實(shí)現(xiàn)被裝飾的抽象接口。
  4. 裝飾器持有被裝飾的抽象接口作為成員屬性。“裝飾”的意思是在原有功能的基礎(chǔ)上擴(kuò)展新功能,因此必須持有原有功能的抽象接口。
  5. 在裝飾器中,對(duì)于需要擴(kuò)展功能的方法,新增擴(kuò)展功能。
  6. 不需要擴(kuò)展功能的方法,直接調(diào)用被裝飾接口的原生方法即可。
  7. 為裝飾器定義一個(gè)工廠方法,入?yún)楸谎b飾接口。
  8. 使用時(shí),通過裝飾器的工廠方法,把所有裝飾器和被裝飾者串聯(lián)起來。

擴(kuò)展

Go 風(fēng)格的實(shí)現(xiàn)

在 Sidecar 的場景上下文中,被裝飾的Socket是一個(gè)相對(duì)復(fù)雜的接口,裝飾器通過實(shí)現(xiàn)Socket接口來進(jìn)行功能擴(kuò)展,是典型的面向?qū)ο箫L(fēng)格。

如果被裝飾者是一個(gè)簡單的接口/方法/函數(shù),我們可以用更具 Go 風(fēng)格的實(shí)現(xiàn)方式,考慮前文提到的 http router 場景。如果你使用原生的net/http進(jìn)行 http router 開發(fā),通常會(huì)這么實(shí)現(xiàn):

funcmain(){
//注冊(cè)/hello的router
http.HandleFunc("/hello",hello)
//啟動(dòng)http服務(wù)器
http.ListenAndServe("localhost:8080",nil)
}

//具體的請(qǐng)求處理邏輯,類型是http.HandlerFunc
funchello(whttp.ResponseWriter,r*http.Request){
w.Write([]byte("hello,world"))
}

其中,我們通過http.HandleFunc來注冊(cè)具體的 router,hello是具體的請(qǐng)求處理方法。現(xiàn)在,我們想為該 http 服務(wù)器增加日志、鑒權(quán)等通用功能,那么可以把func(w http.ResponseWriter, r *http.Request)作為被裝飾的抽象接口,通過新增日志、鑒權(quán)等裝飾器完成功能擴(kuò)展。

//demo/network/http/http_handle_func_decorator.go

//關(guān)鍵點(diǎn)1:確定被裝飾接口,這里為原生的http.HandlerFunc
typeHandlerFuncfunc(ResponseWriter,*Request)

//關(guān)鍵點(diǎn)2:定義裝飾器類型,是一個(gè)函數(shù)類型,入?yún)⒑头祷刂刀际莌ttp.HandlerFunc函數(shù)
typeHttpHandlerFuncDecoratorfunc(http.HandlerFunc)http.HandlerFunc

//關(guān)鍵點(diǎn)3:定義裝飾函數(shù),入?yún)楸谎b飾的接口和裝飾器可變列表
funcDecorate(hhttp.HandlerFunc,decorators...HttpHandlerFuncDecorator)http.HandlerFunc{
//關(guān)鍵點(diǎn)4:通過for循環(huán)遍歷裝飾器,完成對(duì)被裝飾接口的裝飾
for_,decorator:=rangedecorators{
h=decorator(h)
}
returnh
}

//關(guān)鍵點(diǎn)5:實(shí)現(xiàn)具體的裝飾器
funcWithBasicAuth(hhttp.HandlerFunc)http.HandlerFunc{
returnfunc(whttp.ResponseWriter,r*http.Request){
cookie,err:=r.Cookie("Auth")
iferr!=nil||cookie.Value!="Pass"{
w.WriteHeader(http.StatusForbidden)
return
}
//關(guān)鍵點(diǎn)6:完成功能擴(kuò)展之后,調(diào)用被裝飾的方法,才能將所有裝飾器和被裝飾者串起來
h(w,r)
}
}

funcWithLogger(hhttp.HandlerFunc)http.HandlerFunc{
returnfunc(whttp.ResponseWriter,r*http.Request){
log.Println(r.Form)
log.Printf("path%s",r.URL.Path)
h(w,r)
}
}

funchello(whttp.ResponseWriter,r*http.Request){
w.Write([]byte("hello,world"))
}

funcmain(){
//關(guān)鍵點(diǎn)7:通過Decorate函數(shù)完成對(duì)hello的裝飾
http.HandleFunc("/hello",Decorate(hello,WithLogger,WithBasicAuth))
//啟動(dòng)http服務(wù)器
http.ListenAndServe("localhost:8080",nil)
}

上述的裝飾者模式的實(shí)現(xiàn),用到了類似于Functional Options的技巧,也是巧妙利用了 Go 的函數(shù)式編程的特點(diǎn),總結(jié)下來有如下幾個(gè)關(guān)鍵點(diǎn):

  1. 確定被裝飾的接口,上述例子為http.HandlerFunc
  2. 定義裝飾器類型,是一個(gè)函數(shù)類型,入?yún)⒑头祷刂刀际潜谎b飾接口,上述例子為func(http.HandlerFunc) http.HandlerFunc。
  3. 定義裝飾函數(shù),入?yún)楸谎b飾的接口和裝飾器可變列表,上述例子為Decorate方法。
  4. 在裝飾方法中,通過for循環(huán)遍歷裝飾器,完成對(duì)被裝飾接口的裝飾。這里是用來類似Functional Options的技巧,一定要注意裝飾器的順序!
  5. 實(shí)現(xiàn)具體的裝飾器,上述例子為WithBasicAuthWithLogger函數(shù)。
  6. 在裝飾器中,完成功能擴(kuò)展之后,記得調(diào)用被裝飾者的接口,這樣才能將所有裝飾器和被裝飾者串起來。
  7. 在使用時(shí),通過裝飾函數(shù)完成對(duì)被裝飾者的裝飾,上述例子為Decorate(hello, WithLogger, WithBasicAuth)。

Go 標(biāo)準(zhǔn)庫中的裝飾者模式

在 Go 標(biāo)準(zhǔn)庫中,也有一個(gè)運(yùn)用了裝飾者模式的模塊,就是context,其中關(guān)鍵的接口如下:

packagecontext

//被裝飾接口
typeContextinterface{
Deadline()(deadlinetime.Time,okbool)
Done()<-chanstruct{}
Err()error
Value(keyany)any
}

//cancel裝飾器
typecancelCtxstruct{
Context//被裝飾接口
musync.Mutex
doneatomic.Value
childrenmap[canceler]struct{}=
errerror
}
//cancel裝飾器的工廠方法
funcWithCancel(parentContext)(ctxContext,cancelCancelFunc){
//...
c:=newCancelCtx(parent)
propagateCancel(parent,&c)
return&c,func(){c.cancel(true,Canceled)}
}

//timer裝飾器
typetimerCtxstruct{
cancelCtx//被裝飾接口
timer*time.Timer

deadlinetime.Time
}
//timer裝飾器的工廠方法
funcWithDeadline(parentContext,dtime.Time)(Context,CancelFunc){
//...
c:=&timerCtx{
cancelCtx:newCancelCtx(parent),
deadline:d,
}
//...
returnc,func(){c.cancel(true,Canceled)}
}
//timer裝飾器的工廠方法
funcWithTimeout(parentContext,timeouttime.Duration)(Context,CancelFunc){
returnWithDeadline(parent,time.Now().Add(timeout))
}

//value裝飾器
typevalueCtxstruct{
Context//被裝飾接口
key,valany
}
//value裝飾器的工廠方法
funcWithValue(parentContext,key,valany)Context{
ifparent==nil{
panic("cannotcreatecontextfromnilparent")
}
//...
return&valueCtx{parent,key,val}
}
0c708ad6-f700-11ec-ba43-dac502259ad0.jpg

使用時(shí),可以這樣:

//使用時(shí),可以這樣
funcmain(){
ctx:=context.Background()
ctx=context.WithValue(ctx,"key1","value1")
ctx,_=context.WithTimeout(ctx,time.Duration(1))
ctx=context.WithValue(ctx,"key2","value2")
}

不管是 UML 結(jié)構(gòu),還是使用方法,context模塊都與傳統(tǒng)的裝飾者模式有一定出入,但也不妨礙context是裝飾者模式的典型運(yùn)用。還是那句話,學(xué)習(xí)設(shè)計(jì)模式,不能只記住它的結(jié)構(gòu),而是學(xué)習(xí)其中的動(dòng)機(jī)和原理。

典型使用場景

  • I/O 流,比如為原始的 I/O 流增加緩沖、壓縮等功能。
  • Http Router,比如為基礎(chǔ)的 Http Router 能力增加日志、鑒權(quán)、Cookie等功能。
  • ......

優(yōu)缺點(diǎn)

優(yōu)點(diǎn)

  1. 遵循開閉原則,能夠在不修改老代碼的情況下擴(kuò)展新功能。
  2. 可以用多個(gè)裝飾器把多個(gè)功能組合起來,理論上可以無限組合。

缺點(diǎn)

  1. 一定要注意裝飾器裝飾的順序,否則容易出現(xiàn)不在預(yù)期內(nèi)的行為。
  2. 當(dāng)裝飾器越來越多之后,系統(tǒng)也會(huì)變得復(fù)雜。

與其他模式的關(guān)聯(lián)

裝飾者模式和代理模式具有很高的相似性,但是兩種所強(qiáng)調(diào)的點(diǎn)不一樣。前者強(qiáng)調(diào)的是為本體對(duì)象添加新的功能;后者強(qiáng)調(diào)的是對(duì)本體對(duì)象的訪問控制。

裝飾者模式和適配器模式的區(qū)別是,前者只會(huì)擴(kuò)展功能而不會(huì)修改接口;后者則會(huì)修改接口。

文章配圖

可以在用Keynote畫出手繪風(fēng)格的配圖中找到文章的繪圖方法。

審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 模塊
    +關(guān)注

    關(guān)注

    7

    文章

    2655

    瀏覽量

    47292
  • UML
    UML
    +關(guān)注

    關(guān)注

    0

    文章

    122

    瀏覽量

    30839

原文標(biāo)題:【Go實(shí)現(xiàn)】實(shí)踐GoF的23種設(shè)計(jì)模式:裝飾者模式

文章出處:【微信號(hào):yuanrunzi,微信公眾號(hào):元閏子的邀請(qǐng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    【每天學(xué)點(diǎn)AI】一個(gè)例子帶你了解Python裝飾器到底在干嘛!

    今天我們來聊聊一種能給你的代碼變得“加料”的神器——Python裝飾器。就像一杯咖啡,原本它是苦的,為了讓它符合我的口味,我給它添加了糖,添加之后就完美的符合了我的口味。那么,裝飾器又是如何代碼
    的頭像 發(fā)表于 09-20 16:54 ?512次閱讀
    【每天學(xué)點(diǎn)AI】一個(gè)例子帶你了解Python<b class='flag-5'>裝飾</b>器到底在干嘛!

    HarmonyOS實(shí)戰(zhàn)開發(fā)-深度探索與打造個(gè)性化自定義組件

    ,容器組件,媒體組件,繪制組件,畫布組件組件等,如Button、Text 是基礎(chǔ)組件。 由開發(fā)在基礎(chǔ)組件基礎(chǔ)上 添加一些封裝和修飾 定義的組件稱為自定義組件。自定義組件的實(shí)現(xiàn)大大提高
    發(fā)表于 05-08 16:30

    實(shí)踐GoF的23種設(shè)計(jì)模式實(shí)現(xiàn):橋接模式

    也即,將抽象部分和實(shí)現(xiàn)部分進(jìn)行解耦,使得它們能夠各自往獨(dú)立的方向變化。
    的頭像 發(fā)表于 04-14 09:30 ?397次閱讀
    實(shí)踐<b class='flag-5'>GoF</b>的23種設(shè)計(jì)<b class='flag-5'>模式</b>實(shí)現(xiàn):橋接<b class='flag-5'>模式</b>

    鴻蒙OS開發(fā)實(shí)例:【ArkTS類庫多線程@Concurrent裝飾器校驗(yàn)并發(fā)函數(shù)】

    在使用TaskPool時(shí),執(zhí)行的并發(fā)函數(shù)需要使用該裝飾器修飾,否則無法通過相關(guān)校驗(yàn)。從API version 9開始,該裝飾器支持在ArkTS卡片中使用。
    的頭像 發(fā)表于 04-02 14:45 ?564次閱讀
    鴻蒙OS開發(fā)實(shí)例:【ArkTS類庫多線程@Concurrent<b class='flag-5'>裝飾</b>器校驗(yàn)并發(fā)函數(shù)】

    實(shí)踐GoF的23種設(shè)計(jì)模式:解釋器模式

    解釋器模式(Interpreter Pattern)應(yīng)該是 GoF 的 23 種設(shè)計(jì)模式中使用頻率最少的一種了,它的應(yīng)用場景較為局限。
    的頭像 發(fā)表于 04-01 11:01 ?606次閱讀
    實(shí)踐<b class='flag-5'>GoF</b>的23種設(shè)計(jì)<b class='flag-5'>模式</b>:解釋器<b class='flag-5'>模式</b>

    LoRa模組FSK模式下WOR功耗估算實(shí)例

    的LoRa模式,而是使用的該模組的FSK模式,并需要使用FSK模式下的WOR功能。在確定使用FSK模式下的WOR功能時(shí)需要先估算功耗是否能滿足使用
    的頭像 發(fā)表于 03-29 08:12 ?4314次閱讀
    LoRa模組FSK<b class='flag-5'>模式</b>下WOR功耗估算實(shí)例

    鴻蒙OS開發(fā)實(shí)例:【裝飾器@Observed@ObjectLink】

    加深對(duì)@Observed@ObjectLink 裝飾器使用的理解,以小故事做注釋
    的頭像 發(fā)表于 03-28 17:05 ?922次閱讀
    鴻蒙OS開發(fā)實(shí)例:【<b class='flag-5'>裝飾</b>器@Observed@ObjectLink】

    鴻蒙原生應(yīng)用開發(fā)-ArkTS語言基礎(chǔ)類庫多線程@Concurrent裝飾器校驗(yàn)并發(fā)函數(shù)

    在使用TaskPool時(shí),執(zhí)行的并發(fā)函數(shù)需要使用該裝飾器修飾,否則無法通過相關(guān)校驗(yàn)。從API version 9開始,該裝飾器支持在ArkTS卡片中使用。 裝飾器說明 裝飾器使用示例
    發(fā)表于 03-18 10:30

    OpenHarmony父子組件單項(xiàng)同步使用:@Prop裝飾

    @Prop裝飾的變量可以和父組件建立單向的同步關(guān)系。@Prop裝飾的變量是可變的,但是變化不會(huì)同步回其父組件。 說明: 從API version 9開始,該裝飾器支持在ArkTS卡片中使用。 概述
    的頭像 發(fā)表于 02-03 10:57 ?374次閱讀
    OpenHarmony父子組件單項(xiàng)同步使用:@Prop<b class='flag-5'>裝飾</b>器

    OpenHarmony 定義擴(kuò)展組件樣式:@Extend 裝飾

    說明: 從 API version 9 開始,該裝飾器支持在 ArkTS 卡片中使用。 裝飾器使用說明 語法 ? @Extend(UIComponentName) function
    的頭像 發(fā)表于 02-01 20:53 ?249次閱讀

    什么是觀察設(shè)計(jì)模式?Golang中的觀察模式介紹

    當(dāng)涉及到訂單處理系統(tǒng)時(shí),觀察設(shè)計(jì)模式可以用于實(shí)現(xiàn)訂單狀態(tài)的變化和通知。
    的頭像 發(fā)表于 01-08 10:08 ?391次閱讀

    ADE7880除了線周期累計(jì)模式還有其他的累計(jì)模式嗎?兩的區(qū)別是什么?

    您好: 有幾個(gè)問題要請(qǐng)教下: 1、在ADE7880中在關(guān)于電能累計(jì)這一章節(jié)中說到了線周期功率累計(jì)模式,這是一種怎樣的模式? 2、除了線周期累計(jì)模式還有其他的累計(jì)模式嗎?兩
    發(fā)表于 12-27 08:03

    如何C語言中的函數(shù)定義兩個(gè)不同的名字?

    最近有位哥問我,如何C語言中的函數(shù)定義兩個(gè)不同的名字?就是這兩個(gè)名字都是指向同一個(gè)函數(shù),同一個(gè)地址,而且兩個(gè)名字都可以當(dāng)做函數(shù)來用的那種。
    的頭像 發(fā)表于 12-19 16:21 ?715次閱讀

    實(shí)踐GoF的23種設(shè)計(jì)模式:適配器模式

    適配器模式所做的就是將一個(gè)接口 Adaptee,通過適配器 Adapter 轉(zhuǎn)換成 Client 所期望的另一個(gè)接口 Target 來使用,實(shí)現(xiàn)原理也很簡單,就是 Adapter 通過實(shí)現(xiàn) Target接口,并在對(duì)應(yīng)的方法中調(diào)用 Adaptee 的接口實(shí)現(xiàn)。
    的頭像 發(fā)表于 12-10 14:00 ?462次閱讀
    實(shí)踐<b class='flag-5'>GoF</b>的23種設(shè)計(jì)<b class='flag-5'>模式</b>:適配器<b class='flag-5'>模式</b>

    實(shí)踐GoF的23種設(shè)計(jì)模式:備忘錄模式

    相對(duì)于代理模式、工廠模式等設(shè)計(jì)模式,備忘錄模式(Memento)在我們?nèi)粘i_發(fā)中出鏡率并不高,除了應(yīng)用場景的限制之外,另一個(gè)原因,可能是備忘錄模式
    的頭像 發(fā)表于 11-25 09:05 ?506次閱讀
    實(shí)踐<b class='flag-5'>GoF</b>的23種設(shè)計(jì)<b class='flag-5'>模式</b>:備忘錄<b class='flag-5'>模式</b>