Ale*_*lex 47 python setattr getattr python-2.7
我想定义一个包含read和write方法的类,可以按如下方式调用:
instance.read
instance.write
instance.device.read
instance.device.write
Run Code Online (Sandbox Code Playgroud)
为了不使用隔行扫描类,我的想法是覆盖__getattr__和__setattr__方法并检查,如果给定的名称是device将返回重定向到self.但是我遇到了一个无限递归的问题.示例代码如下:
class MyTest(object):
def __init__(self, x):
self.x = x
def __setattr__(self, name, value):
if name=="device":
print "device test"
else:
setattr(self, name, value)
test = MyTest(1)
Run Code Online (Sandbox Code Playgroud)
如在__init__代码中尝试创建一个新属性x,它会调用__setattr__,再次调用__setattr__,依此类推.我怎么需要更改此代码,即,在这种情况下,一个新的属性x的self创建,持有的价值1?
或者有没有更好的方法来处理像instance.device.read'映射'的调用instance.read?
因为总是存在关于原因的问题:我需要创建xmlrpc调用的抽象,为此myxmlrpc.instance,device.read可以创建非常简单的类似方法.我需要'模拟'这样来模仿这样的多点方法调用.
Bak*_*riu 52
您必须调用父类__setattr__方法:
class MyTest(object):
def __init__(self, x):
self.x = x
def __setattr__(self, name, value):
if name=="device":
print "device test"
else:
super(MyTest, self).__setattr__(name, value)
# in python3+ you can omit the arguments to super:
#super().__setattr__(name, value)
Run Code Online (Sandbox Code Playgroud)
关于最佳实践,因为你计划使用这个,xml-rpc我认为这可能在_dispatch方法内部做得更好.
一个快速而肮脏的方法是简单地做:
class My(object):
def __init__(self):
self.device = self
Run Code Online (Sandbox Code Playgroud)
Vin*_*lin 18
或者你可以self.__dict__从里面修改__setattr__():
class SomeClass(object):
def __setattr__(self, name, value):
print(name, value)
self.__dict__[name] = value
def __init__(self, attr1, attr2):
self.attr1 = attr1
self.attr2 = attr2
sc = SomeClass(attr1=1, attr2=2)
sc.attr1 = 3
Run Code Online (Sandbox Code Playgroud)
小智 6
您还可以使用对象。
class TestClass:
def __init__(self):
self.data = 'data'
def __setattr__(self, name, value):
print("Attempt to edit the attribute %s" %(name))
object.__setattr__(self, name, value)
Run Code Online (Sandbox Code Playgroud)