我想完成以下任务:
tmp = do_stuff(tmp2)
if tmp != None:
num = tmp
但是,在我的问题中要分配的变量是对象实例的变量(更好:属性):
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
显然,最后一部分看起来非常重复,在我的情况下需要应用于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
但是这不起作用(因为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
然而,A
,B
和C
是属性,这个计划被挫败了。
类似LISP的宏在这里将是一个巨大的帮助,不幸的是,Python似乎不支持宏,我不想将依赖项添加到我在线找到的Python的一个宏库(我没有' t测试,因为无论如何我都无法使用它们。)
我还想避免在列表中使用匿名函数,因为这可能会产生与第二个代码清单中相同数量的代码。
我解决此问题的最后手段目前是eval
,我想不惜一切代价避免这种情况。
答案 0 :(得分: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)
引用文档:
这是
getattr()
的对应物。参数是一个对象,一个字符串和一个任意值。该字符串可以命名现有属性或新属性。如果对象允许,该函数会将值分配给属性。例如,setattr(x, 'foobar', 123)
相当于x.foobar = 123
。
请注意,由于None
是python中的单例,因此通常使用is
或is 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)
而PEP 8 python styleguide不鼓励复合陈述;将if
套件放在它自己的行上,即使它只是一行。