好的,這是真實的場景:我正在編寫一個應用程序,我有一個類,它表示某種類型的文件(在我的例子中,這是照片,但細節與問題無關)。照片類的每個實例對于照片的文件名都應該是唯一的。
問題是,當用戶告訴我的應用程序加載文件時,我需要能夠識別文件何時已加載,并使用該文件名的現有實例,而不是在同一文件名上創建重復實例。
對我來說,這似乎是一個很好的使用記憶化的環境,有很多這樣的例子,但是在這種情況下,我不僅僅是在記憶一個普通的函數,我需要記憶__init__()。這帶來了一個問題,因為當__init__()被調用時,已經太晚了,因為已經創建了一個新實例。
在我的研究中,我發現了Python的__new__()方法,實際上我可以編寫一個工作簡單的示例,但是當我試圖在我的真實對象上使用它時,它崩潰了,我不知道為什么(我唯一能想到的是,我的真實對象是我無法真正控制的其他對象的子類,因此,這種方法有一些不兼容之處)。這就是我所擁有的:class Flub(object):
instances = {}
def __new__(cls, flubid):
try:
self = Flub.instances[flubid]
except KeyError:
self = Flub.instances[flubid] = super(Flub, cls).__new__(cls)
print 'making a new one!'
self.flubid = flubid
print id(self)
return self
@staticmethod
def destroy_all():
for flub in Flub.instances.values():
print 'killing', flub
a = Flub('foo')
b = Flub('foo')
c = Flub('bar')
print a
print b
print c
print a is b, b is c
Flub.destroy_all()
輸出如下:making a new one!
139958663753808
139958663753808
making a new one!
139958663753872
<__main__.Flub object at 0x7f4aaa6fb050>
<__main__.Flub object at 0x7f4aaa6fb050>
<__main__.Flub object at 0x7f4aaa6fb090>
True False
killing <__main__.Flub object at 0x7f4aaa6fb050>
killing <__main__.Flub object at 0x7f4aaa6fb090>
太完美了!只為給定的兩個唯一id創建了兩個實例,Flub.instances顯然只列出了兩個。
但是當我嘗試對我正在使用的對象使用這種方法時,我得到了各種各樣的荒謬錯誤,關于__init__()如何只使用0個參數,而不是2個參數。所以我會改變一些東西,然后它會告訴我__init__()需要一個參數。完全奇怪。
經過一段時間的斗爭,我基本上放棄了,把所有的__new__()黑魔法移到了一個名為get的staticmethod中,這樣我就可以調用Photograph.get(filename),如果filename不在Photograph.instances中,它只會調用Photograph(filename)。
有人知道我哪里做錯了嗎?有沒有更好的方法來做這個?
另一種思考方式是它類似于單例,只是它不是全局單例,只是每個文件名都是單例。
如果你想一起看的話。