657 Stimmen

Verschachtelte Python-Diktate in Objekte umwandeln?

Ich bin auf der Suche nach einem eleganten Weg, um Daten mit Attribut Zugriff auf ein Diktat mit einigen verschachtelten Dicts und Listen (d.h. Javascript-Stil Objekt-Syntax) zu erhalten.

Zum Beispiel:

>>> d = {'a': 1, 'b': {'c': 2}, 'd': ["hi", {'foo': "bar"}]}

Sollte auf diese Weise zugänglich sein:

>>> x = dict2obj(d)
>>> x.a
1
>>> x.b.c
2
>>> x.d[1].foo
bar

Ich denke, dies ist nicht möglich, ohne Rekursion, aber was wäre ein schöner Weg, um ein Objekt-Stil für Dicts zu erhalten?

6 Stimmen

Ich habe vor kurzem versucht, etwas Ähnliches zu tun, aber ein wiederkehrender Wörterbuchschlüssel ("from" - ein Python-Schlüsselwort) hat mich daran gehindert, es zu tun. Denn sobald man versuchte, mit "x.from" auf dieses Attribut zuzugreifen, erhielt man einen Syntaxfehler.

3 Stimmen

Das ist in der Tat ein Problem, aber ich kann auf "from" verzichten, um das Leben einfacher zu machen, wenn ich auf große Diktatkonstrukte zugreife:) die Eingabe von x['a']['d'][1]['foo'] ist wirklich lästig, also gilt x.a.d[1].foo. wenn Sie from benötigen, können Sie über getattr(x, 'from') darauf zugreifen oder stattdessen _from als Attribut verwenden.

7 Stimmen

from_ statt _from laut PEP 8 .

20voto

user1783597 Punkte 181

Wenn Sie auf Diktatschlüssel als Objekt (oder als Diktat für schwierige Schlüssel) zugreifen möchten, dies rekursiv tun und auch in der Lage sein, das ursprüngliche Diktat zu aktualisieren, können Sie dies tun:

class Dictate(object):
    """Object view of a dict, updating the passed in dict when values are set
    or deleted. "Dictate" the contents of a dict...: """

    def __init__(self, d):
        # since __setattr__ is overridden, self.__dict = d doesn't work
        object.__setattr__(self, '_Dictate__dict', d)

    # Dictionary-like access / updates
    def __getitem__(self, name):
        value = self.__dict[name]
        if isinstance(value, dict):  # recursively view sub-dicts as objects
            value = Dictate(value)
        return value

    def __setitem__(self, name, value):
        self.__dict[name] = value
    def __delitem__(self, name):
        del self.__dict[name]

    # Object-like access / updates
    def __getattr__(self, name):
        return self[name]

    def __setattr__(self, name, value):
        self[name] = value
    def __delattr__(self, name):
        del self[name]

    def __repr__(self):
        return "%s(%r)" % (type(self).__name__, self.__dict)
    def __str__(self):
        return str(self.__dict)

Beispiel für die Verwendung:

d = {'a': 'b', 1: 2}
dd = Dictate(d)
assert dd.a == 'b'  # Access like an object
assert dd[1] == 2  # Access like a dict
# Updates affect d
dd.c = 'd'
assert d['c'] == 'd'
del dd.a
del dd[1]
# Inner dicts are mapped
dd.e = {}
dd.e.f = 'g'
assert dd['e'].f == 'g'
assert d == {'c': 'd', 'e': {'f': 'g'}}

15voto

Anon Punkte 11126
>>> def dict2obj(d):
        if isinstance(d, list):
            d = [dict2obj(x) for x in d]
        if not isinstance(d, dict):
            return d
        class C(object):
            pass
        o = C()
        for k in d:
            o.__dict__[k] = dict2obj(d[k])
        return o

>>> d = {'a': 1, 'b': {'c': 2}, 'd': ["hi", {'foo': "bar"}]}
>>> x = dict2obj(d)
>>> x.a
1
>>> x.b.c
2
>>> x.d[1].foo
'bar'

13voto

Reed Sandberg Punkte 482

In 2021, verwenden Sie pydantic BaseModel - konvertiert verschachtelte dicts und verschachtelte json-Objekte in python-Objekte und umgekehrt:

https://pydantic-docs.helpmanual.io/usage/models/

>>> class Foo(BaseModel):
...     count: int
...     size: float = None
... 
>>> 
>>> class Bar(BaseModel):
...     apple = 'x'
...     banana = 'y'
... 
>>> 
>>> class Spam(BaseModel):
...     foo: Foo
...     bars: List[Bar]
... 
>>> 
>>> m = Spam(foo={'count': 4}, bars=[{'apple': 'x1'}, {'apple': 'x2'}])

Objekt zum Diktat

>>> print(m.dict())
{'foo': {'count': 4, 'size': None}, 'bars': [{'apple': 'x1', 'banana': 'y'}, {'apple': 'x2', 'banana': 'y'}]}

Objekt zu JSON

>>> print(m.json())
{"foo": {"count": 4, "size": null}, "bars": [{"apple": "x1", "banana": "y"}, {"apple": "x2", "banana": "y"}]}

Diktat zu Objekt

>>> spam = Spam.parse_obj({'foo': {'count': 4, 'size': None}, 'bars': [{'apple': 'x1', 'banana': 'y'}, {'apple': 'x2', 'banana': 'y2'}]})
>>> spam
Spam(foo=Foo(count=4, size=None), bars=[Bar(apple='x1', banana='y'), Bar(apple='x2', banana='y2')])

JSON zu Objekt

>>> spam = Spam.parse_raw('{"foo": {"count": 4, "size": null}, "bars": [{"apple": "x1", "banana": "y"}, {"apple": "x2", "banana": "y"}]}')
>>> spam
Spam(foo=Foo(count=4, size=None), bars=[Bar(apple='x1', banana='y'), Bar(apple='x2', banana='y')])

0 Stimmen

Dies sollte die akzeptierte Antwort sein. Ich danke Ihnen vielmals! Ich habe eine Stunde damit verbracht, nach einer Lösung zu suchen, die gut mit Typ-annotierten Klassen funktioniert.

0 Stimmen

Dies funktioniert zwar, wenn Sie die Struktur des Diktats kennen, aber Sie können kein pydantisches Modell dynamisch erstellen. Dazu müssen Sie die Funktion create_model(__model_name="ModelName", **some_dict) . Dies funktioniert jedoch nur mit der obersten Ebene, so dass Sie es dazu bringen müssen, pydantische Modelle rekursiv zu erzeugen. Aber die Erstellung von pydantischen Modellen ist langsamer als andere Module wie named_tuples oder dataclasses.

0 Stimmen

Wie groß ist der Aufwand von pydantic im Vergleich zu munch im Vergleich zu namedtuples?

11voto

Alex Rodrigues Punkte 2237

x.__dict__.update(d) sollte gut funktionieren.

0 Stimmen

Danke für Ihre Antwort, aber was ist x? das Diktat oder ein Standardobjekt? könnten Sie mir bitte einen Hinweis geben.

0 Stimmen

X ist Ihr Objekt. Jedes Objekt hat eine Diktat . Durch die Aktualisierung der Diktat eines Objekts aktualisieren Sie tatsächlich die Schlüsselvariablen in diesem Objekt.

0 Stimmen

Die fettgedruckten Wörter sind _ _ dict _ _

9voto

VengaVenga Punkte 378

Normalerweise möchten Sie die Diktat-Hierarchie in Ihrem Objekt spiegeln, nicht aber Listen oder Tupel, die sich normalerweise auf der untersten Ebene befinden. So habe ich das also gemacht:

class defDictToObject(object):

    def __init__(self, myDict):
        for key, value in myDict.items():
            if type(value) == dict:
                setattr(self, key, defDictToObject(value))
            else:
                setattr(self, key, value)

Also tun wir es:

myDict = { 'a': 1,
           'b': { 
              'b1': {'x': 1,
                    'y': 2} },
           'c': ['hi', 'bar'] 
         }

und erhalten:

x.b.b1.x 1

x.c ['hi', 'bar']

CodeJaeger.com

CodeJaeger ist eine Gemeinschaft für Programmierer, die täglich Hilfe erhalten..
Wir haben viele Inhalte, und Sie können auch Ihre eigenen Fragen stellen oder die Fragen anderer Leute lösen.

Powered by:

X