如何正确使用__setattr__,避免无限递归

Ale*_*lex 47 python setattr getattr python-2.7

我想定义一个包含readwrite方法的类,可以按如下方式调用:

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__,依此类推.我怎么需要更改此代码,即,在这种情况下,一个新的属性xself创建,持有的价值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)

  • 因为这不是设置 `__dict__` 属性,而是先获取它然后在返回的 dict 上设置一个项目,即,它正在调用诸如:`getattr(self, "__dict__").__setitem__(name, value)` (4认同)

小智 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)