以编程方式设置属性

use*_*579 2 python setter properties

我想完成以下任务:

tmp = do_stuff(tmp2)
if tmp != None:
    num = tmp
Run Code Online (Sandbox Code Playgroud)

但是,我的问题中要分配的变量是对象实例的变量(更好:属性):

class mycls(object):
    def __init__(self):
        self.__a, self._b, self.c = 2, 3, 5
    def set_a(self, val): self.__a = val
    def set_b(self, val): self._b = val
    def set_c(self, val): self.c = val
    A = property(fset = set_a)
    B = property(fset = set_b)
    C = property(fset = set_c)
tmp2 = [7, 11, 13]
inst = mycls()
tmp = do_stuff(tmp2[0])
if tmp != None: inst.A = tmp
tmp = do_stuff(tmp2[1])
if tmp != None: inst.B = tmp
tmp = do_stuff(tmp2[2])
if tmp != None: inst.C = tmp
Run Code Online (Sandbox Code Playgroud)

显然,最后一部分看起来高度重复,在我的情况下需要应用于10多个变量.我现在的问题是:如何压缩最后6行?

自己尝试解决问题

理想情况下,我更喜欢这样的解决方案:

for i in [[inst.A, 0], [inst.B, 1], [inst.C, 2]]:
    tmp = do_stuff(tmp2[i[1]])
    if tmp != None: i[0] = tmp
Run Code Online (Sandbox Code Playgroud)

但是这不起作用(因为inst.X已经过评估,您无法通过引用/指针存储原始数据类型).我想出的另一种方法是使用变量名称作为字符串并使用inst.__dict__,例如:

for i in [["A", 0], ["B", 1], ["C", 2]]:
    tmp = do_stuff(tmp2[i[1]])
    if tmp != None: inst.__dict__[i[0]] = tmp
Run Code Online (Sandbox Code Playgroud)

然而,这个计划被这个事实所挫败A,B并且C是属性.

类似LISP的宏在这里将是一个巨大的帮助,不幸的是Python似乎不支持宏,我不想将依赖项添加到我在线找到的Python的一个宏库(我之前没有测试过)反正我也无法使用它们.

我还想避免在列表中使用匿名函数,因为这可能会产生与第二个代码清单中相同数量的代码.

我最后解决这个问题的方法目前是evals,我想不惜一切代价避免这种情况.

Mar*_*ers 6

您正在寻找setattr():

for i in [["A", 0], ["B", 1], ["C", 2]]:
    tmp = do_stuff(tmp2[i[1]])
    if tmp != None: setattr(inst, i[0], tmp)
Run Code Online (Sandbox Code Playgroud)

引用文档:

这是对应的getattr().参数是一个对象,一个字符串和一个任意值.该字符串可以命名现有属性或新属性.如果对象允许,该函数会将值分配给属性.例如,setattr(x, 'foobar', 123)相当于x.foobar = 123.

注意,因为None在python中是单例,通常使用isis not运算符测试它:

for i in [["A", 0], ["B", 1], ["C", 2]]:
    tmp = do_stuff(tmp2[i[1]])
    if tmp is not None:
        setattr(inst, i[0], tmp)
Run Code Online (Sandbox Code Playgroud)

并且PEP 8 python样式指南不鼓励复合语句; 将if套件保持在自己的线上,即使它只是一条线.