我想在Python中创建一个动态对象(在另一个对象内部),然后向其添加属性。
我试着:
obj = someobject
obj.a = object()
setattr(obj.a, 'somefield', 'somevalue')
但这并不奏效。
什么好主意吗?
编辑:
我设置的属性从一个循环通过一个值的列表,例如。
params = ['attr1', 'attr2', 'attr3']
obj = someobject
obj.a = object()
for p in params:
obj.a.p # where p comes from for loop variable
在上面的例子中,我会得到obj.a。attr1, obj.a。attr2 obj.a.attr3。
我使用了setattr函数,因为我不知道如何从for循环中执行obj.a.NAME。
如何根据上面例子中的p值设置属性?
有关为什么它不起作用的详细信息,请参见不能在“object”类的实例上设置属性。
内置对象可以被实例化,但不能在其上设置任何属性。(为了这个目的,我希望它可以。)它没有__dict__来保存属性。
我通常是这样做的:
class Object(object):
pass
a = Object()
a.somefield = somevalue
当我可以时,我给Object类起一个更有意义的名字,这取决于我在其中放入什么样的数据。
有些人做了不同的事情,他们使用dict的子类,允许属性访问来获得键。(d.key而不是d['key'])
编辑:对于您的问题,使用setattr是很好的。你不能在object()实例上使用setattr。
params = ['attr1', 'attr2', 'attr3']
for p in params:
setattr(obj.a, p, value)
您正在使用哪些对象?刚刚尝试了一个样例类,它工作得很好:
class MyClass:
i = 123456
def f(self):
return "hello world"
b = MyClass()
b.c = MyClass()
setattr(b.c, 'test', 123)
b.c.test
结果是123。
我看到这个失败的唯一情况是如果你在一个内置对象上尝试setattr。
更新:从评论这是一个重复:为什么你不能添加属性的对象在python?
试试下面的代码:
$ python
>>> class Container(object):
... pass
...
>>> x = Container()
>>> x.a = 10
>>> x.b = 20
>>> x.banana = 100
>>> x.a, x.b, x.banana
(10, 20, 100)
>>> dir(x)
['__class__', '__delattr__', '__dict__', '__doc__', '__format__',
'__getattribute__', '__hash__', '__init__', '__module__', '__new__',
'__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__',
'__str__', '__subclasshook__', '__weakref__', 'a', 'b', 'banana']
有不同的类型。Python 3.3+中的SimpleNamespace类:
obj = someobject
obj.a = SimpleNamespace()
for p in params:
setattr(obj.a, p, value)
# obj.a.attr1
collections.namedtuple,打字。NamedTuple可用于不可变对象。PEP 557—数据类建议一个可变的替代方案。
要获得更丰富的功能,可以尝试attrs包。参见示例用法。Pydantic可能也值得一看。
今天晚了,但这是我的一文不值的一个对象,恰好在应用程序中保存了一些有用的路径,但你可以将它用于任何你想要的信息字典,你可以通过getattr和点符号访问(这是我认为这个问题真正的问题):
import os
def x_path(path_name):
return getattr(x_path, path_name)
x_path.root = '/home/x'
for name in ['repository', 'caches', 'projects']:
setattr(x_path, name, os.path.join(x_path.root, name))
这很酷,因为:
In [1]: x_path.projects
Out[1]: '/home/x/projects'
In [2]: x_path('caches')
Out[2]: '/home/x/caches'
所以这使用函数对象像上面的答案,但使用函数来获取值(你仍然可以使用(getattr, x_path, 'repository'),而不是x_path('repository'),如果你喜欢的话)。
如果我们可以在创建嵌套对象之前确定并聚合所有属性和值,那么我们可以创建一个新类,在创建时接受dictionary参数。
# python 2.7
class NestedObject():
def __init__(self, initial_attrs):
for key in initial_attrs:
setattr(self, key, initial_attrs[key])
obj = someobject
attributes = { 'attr1': 'val1', 'attr2': 'val2', 'attr3': 'val3' }
obj.a = NestedObject(attributes)
>>> obj.a.attr1
'val1'
>>> obj.a.attr2
'val2'
>>> obj.a.attr3
'val3'
我们还可以允许关键字参数。请看这篇文章。
class NestedObject(object):
def __init__(self, *initial_attrs, **kwargs):
for dictionary in initial_attrs:
for key in dictionary:
setattr(self, key, dictionary[key])
for key in kwargs:
setattr(self, key, kwargs[key])
obj.a = NestedObject(attr1='val1', attr2='val2', attr3= 'val3')
如果你正在寻找链赋值,要做的事情,如django模型模板抽象属性赋值:
from types import SimpleNamespace
def assign(target, *args, suffix):
ls = target
for i in range(len(args) - 1):
a = args[i]
ns = SimpleNamespace()
setattr(ls, a, ns)
ls = ns
setattr(ls, args[-1], suffix)
return ls
a = SimpleNamespace()
assign(a, 'a', 'b', 'c', suffix={'name': 'james'})
print(a.a.b.c)
# {'name': 'james'}
它允许您传递模型作为目标,并分配结束属性给它。