Keine direkte Antwort auf die Frage des OP, aber inspiriert von und vielleicht nützlich für einige.. Ich habe eine objektbasierte Lösung mit dem internen __dict__
erstellt (In keiner Weise optimierter Code)
payload = {
"name": "John",
"location": {
"lat": 53.12312312,
"long": 43.21345112
},
"numbers": [
{
"role": "home",
"number": "070-12345678"
},
{
"role": "office",
"number": "070-12345679"
}
]
}
class Map(object):
"""
Punktzugriff auf Objektmitglieder, Zugriff auf Rohwerte
mit einem Unterstrich z.B.
class Foo(Map):
def foo(self):
return self.get('foo') + 'bar'
obj = Foo(**{'foo': 'foo'})
obj.foo => 'foobar'
obj._foo => 'foo'
"""
def __init__(self, *args, **kwargs):
for arg in args:
if isinstance(arg, dict):
for k, v in arg.items():
self.__dict__[k] = v
self.__dict__['_' + k] = v
if kwargs:
for k, v in kwargs.items():
self.__dict__[k] = v
self.__dict__['_' + k] = v
def __getattribute__(self, attr):
if hasattr(self, 'get_' + attr):
return object.__getattribute__(self, 'get_' + attr)()
else:
return object.__getattribute__(self, attr)
def get(self, key):
try:
return self.__dict__.get('get_' + key)()
except (AttributeError, TypeError):
return self.__dict__.get(key)
def __repr__(self):
return u"<{name} object>".format(
name=self.__class__.__name__
)
class Number(Map):
def get_role(self):
return self.get('role')
def get_number(self):
return self.get('number')
class Location(Map):
def get_latitude(self):
return self.get('lat') + 1
def get_longitude(self):
return self.get('long') + 1
class Item(Map):
def get_name(self):
return self.get('name') + " Doe"
def get_location(self):
return Location(**self.get('location'))
def get_numbers(self):
return [Number(**n) for n in self.get('numbers')]
# Tests
obj = Item({'foo': 'bar'}, **payload)
assert type(obj) == Item
assert obj._name == "John"
assert obj.name == "John Doe"
assert type(obj.location) == Location
assert obj.location._lat == 53.12312312
assert obj.location._long == 43.21345112
assert obj.location.latitude == 54.12312312
assert obj.location.longitude == 44.21345112
for n in obj.numbers:
assert type(n) == Number
if n.role == 'home':
assert n.number == "070-12345678"
if n.role == 'office':
assert n.number == "070-12345679"
32 Stimmen
Viele der Situationen, in denen Menschen verschachtelte Dicts verwenden, könnten genauso gut oder sogar besser durch Dicts mit Tupeln als Schlüssel gelöst werden, wobei
d[a][b][c]
durchd[a, b, c]
ersetzt wird.0 Stimmen
Können Sie bitte näher erläutern, wie
d[a][b][c]
durchd[a, b, c]
ersetzt werden kann? Ich habe noch nicht verstanden, wie das gemacht werden kann.9 Stimmen
Es ist kein Zauberei: foo={}; foo[1,2,3] = "one, two, three!"; foo.keys() => [(1,2,3)]
16 Stimmen
Wow. Wow noch einmal. Ich wusste nicht, dass Tupel Schlüssel von Dict sein können. Wow zum dritten Mal.
5 Stimmen
Jedes Objekt, das "hashbar" ist, kann als Schlüssel eines Dicts verwendet werden. Die meisten unveränderlichen Objekte sind auch hashbar, aber nur, wenn alle ihre Inhalte hashbar sind. Der Code d[1, 2, 3] funktioniert, weil "," der "Erstellen eines Tupeloperators" ist; es ist dasselbe wie d[(1, 2, 3)]. Klammern sind häufig optional bei der Deklaration eines Tupels.
0 Stimmen
stackoverflow.com/questions/3031219/… hat eine Antwort, die mehr macht, als Sie wollen.
1 Stimmen
Eine Situation, in der ein Tupel nicht genauso gut ist, json.dumps, TypeError: Schlüssel müssen eine Zeichenfolge sein
0 Stimmen
pypi.python.org/pypi/easydict
0 Stimmen
Bitte sehen Sie auch stackoverflow.com/questions/4984647/…
7 Stimmen
Hast du den Fall in Betracht gezogen, in dem der Schlüssel nur einen Punkt für sich hat -
{"my.key":"value"}
? Oder wenn der Schlüssel ein Schlüsselwort ist, wie z.B. "from"? Ich habe es ein paar Mal in Betracht gezogen, und es bringt mehr Probleme und Fehlerbehebung als wahrgenutzte Vorteile.1 Stimmen
Wenn Sie nicht dazu verpflichtet sind, dict zu verwenden, können Sie stattdessen namedtuple (docs.python.org/2/library/…) verwenden - auf diese Weise können Sie auf seine Elemente über den Punkt (".") zugreifen.
0 Stimmen
Dieser Frage wird auf eine sehr ähnliche Art und Weise vernetzt: Wie man Punkt Notation für ein Wörterbuch in Python verwendet? Diese hat interessante Antworten unter Verwendung von
SimpleNamespace
aus dem integriertentypes
Modul.0 Stimmen
Siehe auch: stackoverflow.com/questions/23463684/…
1 Stimmen
@TodorMinakov einen Punkt in einem Wörterbuchschlüssel zu haben ist genauso üblich wie das Codieren auf einem Bildschirm, der auf den Kopf gestellt ist - und genauso einfach umkehrbar. Außerdem löst es keinerlei Fehler aus, wenn ein Objekt ein
from_whatever
-Attribut hat.