AttributeError:无法在python中设置属性

Pra*_*uka 59 python attributes

这是我的代码

N = namedtuple("N", ['ind', 'set', 'v'])
def solve()
    items=[]
    stack=[]
    R = set(range(0,8))
    for i in range(0,8):
        items.append(N(i,R,8))      
        stack.append(N(0,R-set(range(0,1)),i))
    while(len(stack)>0): 
        node = stack.pop()
        print node
        print items[node.ind]   
        items[node.ind].v = node.v
Run Code Online (Sandbox Code Playgroud)

在最后一行,我不能将items[node.ind].v值设置node.v为我想要的,并得到错误

"AttributeError: can't set attribute"
Run Code Online (Sandbox Code Playgroud)

我不知道什么是错的,但它必须是基于语法的东西,因为使用语句node.v+=1也会显示相同的错误.我是Python的新手,所以请建议一种方法来实现上述变更.

jon*_*rpe 59

namedtuples是不可变的,就像标准元组一样.你有两个选择:

  1. 使用不同的数据结构,例如类(或只是字典); 要么
  2. 而不是更新结构,替换它.

前者看起来像:

class N(object):

    def __init__(self, ind, set, v):
        self.ind = ind
        self.set = set
        self.v = v
Run Code Online (Sandbox Code Playgroud)

而后者:

item = items[node.ind]
items[node.ind] = N(item.ind, item.set, node.v)
Run Code Online (Sandbox Code Playgroud)

编辑:如果你想要后者,Ignacio的答案使用烘焙功能更加巧妙地做同样的事情.

  • namedtuples 是不可变的 - 一针见血。谢谢 (4认同)
  • “使用不同的数据结构,例如类”——可能是[@dataclass](https://docs.python.org/3/library/dataclasses.html)? (3认同)

Ign*_*ams 54

items[node.ind] = items[node.ind]._replace(v=node.v)
Run Code Online (Sandbox Code Playgroud)

(注意:不要因为函数_replace中的前导下划线而不鼓励使用此解决方案.特别是对于namedtuple,某些函数具有前导下划线,而不是用于表示它们是"私有")

  • @ IgnacioVazquez-Abrams在这个答案中需要更多解释.(链接到文档是不够的.) (14认同)
  • 为什么呢?⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ (10认同)
  • [由于](https://docs.python.org/3/library/collections.html#collections.somenamedtuple._replace) (7认同)
  • TLDR; 这是一个有效的解决方案.函数_replace中的前导下划线并不表示内部使用. (7认同)
  • @kroiz:“有一个原因......”是的,有,就是这样:[“为了防止与字段名称冲突,方法和属性名称以下划线开头。”](https://docs. python.org/3/library/collections.html#collections.somenamedtuple._replace) (3认同)
  • @kroiz Dennis说我们使用下划线来表示方法和属性以区别于字段名称,jonrsharpe给出了详细的答案,但我的问题是找到纠正错误的最佳方法,使用内置功能总是更好..甚至jonrsharpe提到那部分..坦率地说,我会接受这两个但堆栈不允许 (2认同)

Azm*_*sov 22

对于那些搜索这个错误的人来说,AtributeError: can't set attribute如果你尝试设置一个@property没有 setter 方法的装饰,可以触发的另一件事是。不是 OP 问题中的问题,但我把它放在这里是为了帮助直接搜索错误消息。(如果你不喜欢它,去编辑问题的标题:)

class Test:
    def __init__(self):
        self._attr = "original value"
        # This will trigger an error...
        self.attr = "new value"
    @property
    def attr(self):
        return self._attr

Test()
Run Code Online (Sandbox Code Playgroud)