0
import attr
@attr.s(slots=True, frozen=True)
class C:
    url = attr.ib(type=str)
    x = attr.ib()
    y = attr.ib()
    z = attr.ib(default=10)
    
    @y.default
    def _any_name_except_a_name_of_an_attribute(self):
        return self.x + 1
    
    @url.validator
    def map_url(self, attribute, value):
        if value == "apple":
            self.url = "mango"

print(C(x=4,y=5, url="apple"))

我希望找到一种在初始化为特定值时更改 url 的方法,而不会丢失类的冻结属性,这是否可能?

4

2 回答 2

2

更改传入的值是转换器的用途:

url = attr.ib(type=str, converter=lambda x: 'mango' if x=='apple' else x)
于 2020-07-20T17:53:19.030 回答
0

或者,这可以使用attr.evolve方法在初始化后实现,该方法创建实例的副本,并对其应用指定的更改,从而保持对类实例的冻结约束。

顺便说一句,这种对不可变实例的修改在函数式编程中随处可见,例如 scala 的案例类复制方法

>>> c = C(x=4, y=5, url="apple")

# create a new instance with a different url
>>> attr.evolve(c, url="banana")
C(url='banana', x=4, y=5, z=10)

# original instance remains unchanged
>>> c
C(url='apple', x=4, y=5, z=10)
于 2022-01-13T05:21:42.073 回答