我试图想出一种方法来跟踪Python中对象的各种重新实现。例如,我想为一个房间开设一个课程。这个房间可以作为一个“厨房”开始存在,然后,在任何给定的时间点,成为一个“研究”。我希望能够跟踪(或记录)我房间生活中的所有这些不同阶段。所以,我想出了这个,我想知道是否有更好的方法来做到这一点。或者也许我错过了什么。但据我所知,它似乎适用于我的目的。
class room():
def __init__(self,current="Any", history = []):
self.current = childroom(current)
self.history = history
self.history.append(self.current)
def changeroom(self,current = "New"):
self.current = childroom(current)
self.history.append(self.current)
class childroom(room):
def __init__(self,name="Any"):
self.name = name
当我使用此代码时......
>>> myroom = room("kitchen")
>>> myroom.changeroom("study")
>>> myroom
<__main__.room instance at 0x0000000003139208>
>>> myroom.history
[<__main__.childroom instance at 0x0000000003139288>, <__main__.childroom instance at 0x00000000023F4AC8>]
>>> myroom.history[0].name
'kitchen'
答案 0 :(得分:2)
我个人会像这样实现它:
#! /usr/bin/python3
import copy
class Room:
def __init__ (self, state = 'Initial'):
self.state = state
self.history = []
def morph (self, state):
clone = copy.deepcopy (self)
self.state = state
self.history.append (clone)
请注意,我不知道您的真实设置是否具有某些限制深度复制的功能。
这会产生:
>>> r = Room ('Kitchen')
>>> r.morph ('Loo')
>>> r.morph ('Spaceship')
>>> r.state
'Spaceship'
>>> [a.state for a in r.history]
['Kitchen', 'Loo']
>>> [type (a) for a in r.history]
[<class 'test.Room'>, <class 'test.Room'>]
我猜你通常不需要保存对象的整个状态,只需要保存值得跟踪的属性。您可以沿着这些行将此行为打包到装饰器中:
#! /usr/bin/python3
import datetime
import copy
class Track:
def __init__ (self, *args, saveState = False):
self.attrs = args
self.saveState = saveState
def __call__ (self, cls):
cls._track = []
this = self
oGetter = cls.__getattribute__
def getter (self, attr):
if attr == 'track': return self._track
if attr == 'trackTrace': return '\n'.join ('{}: "{}" has changed to "{}"'.format (*t) for t in self._track)
return oGetter (self, attr)
cls.__getattribute__ = getter
oSetter = cls.__setattr__
def setter (self, attr, value):
if attr in this.attrs:
self._track.append ( (datetime.datetime.now (), attr, copy.deepcopy (value) if this.saveState else value) )
return oSetter (self, attr, value)
cls.__setattr__ = setter
return cls
现在我们可以像这样使用这个装饰器:
@Track ('holder')
class Book:
def __init__ (self, title):
self.title = title
self.holder = None
self.price = 8
class Person:
def __init__ (self, firstName, lastName):
self.firstName = firstName
self.lastName = lastName
def __str__ (self):
return '{} {}'.format (self.firstName, self.lastName)
r = Book ('The Hitchhiker\'s Guide to the Galaxy')
p = Person ('Pedro', 'Párramo')
q = Person ('María', 'Del Carmen')
r.holder = p
r.price = 12
r.holder = q
q.lastName = 'Del Carmen Orozco'
print (r.trackTrace)
如果使用@Track ('holder')
进行调用,则会产生:
2013-10-01 14:02:43.748855: "holder" has changed to "None"
2013-10-01 14:02:43.748930: "holder" has changed to "Pedro Párramo"
2013-10-01 14:02:43.748938: "holder" has changed to "María Del Carmen Orozco"
如果使用@Track ('holder', 'price')
进行调用,则会产生:
2013-10-01 14:05:59.433086: "holder" has changed to "None"
2013-10-01 14:05:59.433133: "price" has changed to "8"
2013-10-01 14:05:59.433142: "holder" has changed to "Pedro Párramo"
2013-10-01 14:05:59.433147: "price" has changed to "12"
2013-10-01 14:05:59.433151: "holder" has changed to "María Del Carmen Orozco"
如果使用@Track ('holder', saveState = True)
进行调用,则会产生:
2013-10-01 14:06:36.815674: "holder" has changed to "None"
2013-10-01 14:06:36.815710: "holder" has changed to "Pedro Párramo"
2013-10-01 14:06:36.815779: "holder" has changed to "María Del Carmen"
答案 1 :(得分:0)
这听起来像是一个外观模式的情况 - 有一个内部空间对象捕获当前状态,并在立面上使用__setattr__
来捕获任务,存档以前的状态(内部对象),并创建一个新的一个具有更新的属性值。您将要覆盖__getattr__
以委托给内部对象,内部对象也将具有适用于其他任何内容的方法。