получение динамического атрибута в python

У меня есть и объект с псевдо или специальным атрибутом, который можно назвать тремя различными способами (Примечание: я не управляю кодом, который генерирует объект)

Значение в атрибутах (в зависимости от того, какой из них задано) является точно таким же, и мне нужно получить его для дальнейшей обработки, поэтому в зависимости от источника данных у меня может быть что-то вроде:

>>> obj.a
'value'
>>> obj.b
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: Obj instance has no attribute 'b'
>>> obj.c
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: Obj instance has no attribute 'c'

или

>>> obj.a
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: Obj instance has no attribute 'a'
>>> obj.b
'value'
>>> obj.c
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: Obj instance has no attribute 'c'

или

>>> obj.a
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: Obj instance has no attribute 'a'
>>> obj.b
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: Obj instance has no attribute 'b'
 >>> obj.c
'value'

Я заинтересован в получении 'value' и, к сожалению, свойство __dict__ не существует в этом объекте. Поэтому то, что я закончил делать для получения этой ценности, было просто кучей вызовов getattr. Предполагая, что возможностей всего три, код выглядит так:

>>> g = lambda o, l: getattr(o, l[0], getattr(o, l[1], getattr(o, l[2], None)))
>>> g(obj, ('a', 'b', 'c'))
'value'

Теперь я хотел бы знать, есть ли лучший способ для этого? Поскольку я на 100% убежден, что я сделал :)

заранее спасибо

Ответ 1

Как насчет:

for name in 'a', 'b', 'c':
    try:
        thing = getattr(obj, name)
    except AttributeError:
        pass
    else:
        break

Ответ 2

Это имеет преимущество в работе с любым количеством элементов:

 def getfirstattr(obj, *attrs):
     return next((getattr(obj, attr) for attr in attrs 
                  if hasattr(obj, attr)), None)

У этого есть очень незначительный недостаток, что он выполняет два поиска по окончательному значению: один раз, чтобы проверить, существует ли атрибут, другой - фактически получить значение. Этого можно избежать, используя выражение вложенного генератора:

 def getfirstattr(obj, *attrs):
     return next((val for val in (getattr(obj, attr, None) for attr in attrs)
                  if val is not None), None)

Но я действительно не считаю это большим делом. Выражение генератора, вероятно, будет быстрее, чем простой старый цикл, даже с двойным поиском.

Ответ 3

Я думаю, используя dir, вы получите по существу то же самое, __dict__ обычно __dict__...

targetValue = "value"
for k in dir(obj):
    if getattr(obj,k) == targetValue:
       print "%s=%s"%(k,targetValue)

что-то вроде

>>> class x:
...    a = "value"
...
>>> dir(x)
['__doc__', '__module__', 'a']
>>> X = x()
>>> dir(X)
['__doc__', '__module__', 'a']
>>> for k in dir(X):
...     if getattr(X,k) == "value":
...        print "%s=%s"%(k,getattr(X,k))
...
a=value
>>>

Ответ 4

Там может быть лучший способ сделать это в зависимости от структуры вашего объекта, но ничего не зная, вот рекурсивное решение, которое работает точно так же, как ваше текущее решение, за исключением того, что оно будет работать с произвольным числом аргументов:

g = lambda o, l: getattr(o, l[0], g(o, l[1:])) if l else None

Ответ 5

И еще один:

reduce(lambda x, y:x or  getattr(obj, y, None),  "a b c".split(), None)

(в Python 3 вам нужно импортировать сокращение из functools. Это встроенный в Python 2)