生活随笔
收集整理的這篇文章主要介紹了
Python的弱引用
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
python的弱引用指引用一個對象但不增加它的引用計數器。這么做的好處是什么呢?什么時候需要考慮用若引用呢?
假設我們在設計一個游戲,有一個角色類Char,我們要給他添加一個效果(比如中毒),于是設計了一個效果類Effect?,F在,給角色增加效果看上去就像這樣:
char.effect?=?Effect()?# 給角色添加一個效果 每個效果生效的時機都是不同的,為了方便復用,我們再設計一個激活策略類ActivePloy,負責激活效果。于是在Effect和ActivePloy的內部看上去就像這樣:
class?Effect(object):???????def?__init__(self):???????????self.active_ploy?=?ActivePloy(self)??????def?active(self):??????????????????pass???class?ActivePloy(object):?????def?__init__(self,?effect):?????????self.effect?=?effect??????def?active(self):??????????????????self.effect.active()? 這樣做的好處是Effect不用關心自己何時激活,激活的判斷都放給ActivePloy來處理。看上去挺好的,但是,這里面有一個問題,就是當我們試圖給玩家去掉這個效果時……
del?char.effect? 仔細想想,這么干以后,Effect的實例其實是沒有被回收的,因為Effect和ActivePloy交叉引用,他們的引用計數都為1。
那么我們為了干凈的刪除effect,似乎就只能手動的來清理一下他們之間的這個交叉引用了:
class?Effect(object):????????def?__init__(self):????????????self.active_ploy?=?ActivePloy(self)????????def?active(self):????????????????????pass????????def?destroy(self):?????????self.active_ploy.destroy()???class?ActivePloy(object):??????def?__init__(self,?effect):??????????self.effect?=?effect????????def?active(self):????????????????????self.effect.active()??????def?destroy(self):?????????self.effect?=?None? 于是我們要刪除一個效果,就得這樣:
char.effect.destroy()?del?char.effect? 太麻煩了,不是嗎?而且萬一一個效果有多個激活策略的話,必須保證Effect把每個ActivePloy的destroy方法都運行一遍,漏了一個都無法保證自身被干凈的刪除。
我們來分析一下,之所以這么麻煩,就是因為ActivePloy對Effect有一個引用。那么如果ActivePloy不引用Effect不就OK了?這個時候,讓我們來試試弱引用。
import?weakref?class?Effect(object):????????def?__init__(self):????????????self.active_ploy?=?ActivePloy(self)????????def?active(self):????????????????????pass??????class?ActivePloy(object):??????def?__init__(self,?effect):??????????self.effect?=?weakref.proxy(effect)????????def?active(self):????????????????????self.effect.active()? 代碼只有一個地方改變了,就是
self.effect?=?weakref.proxy(effect)? 這句的效果就是self.effect可以像往常一樣的使用,但是卻不會增加effect的引用計數器。換言之,這樣寫,他們之間的交叉關系消失了!這個時候我們只需要單純的刪掉char.effect,Effect和ActivePloy的實例都會被銷毀。
什么,假設ActivePloy在其他地方也被引用了?這樣當然只有effect會被銷毀。但是我們想讓ActivePloy必然隨著Effect的銷毀而銷毀,怎么辦呢?那么我們可以改改,給弱引用加上一個回調函數:
class?ActivePloy(object):???????def?__init__(self,?effect):???????????self.effect?=?weakref.proxy(effect,?self.on_effect_destroy)?????????def?active(self):??????????????????????self.effect.active()???????def?on_effect_destroy(self,?effect):????????????????????pass? 這樣一來,就不用擔心刪不干凈了。
轉載于:https://blog.51cto.com/sleepd/1073044
創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎
總結
以上是生活随笔為你收集整理的Python的弱引用的全部內容,希望文章能夠幫你解決所遇到的問題。
如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。