Ponieważ wydaje się, że nikt o tym nie wspomniał deepdiff
, dodam go tutaj dla kompletności. Uważam, że bardzo wygodne jest uzyskiwanie różnych (zagnieżdżonych) obiektów w ogóle:
Instalacja
pip install deepdiff
Przykładowy kod
import deepdiff
import json
dict_1 = {
"a": 1,
"nested": {
"b": 1,
}
}
dict_2 = {
"a": 2,
"nested": {
"b": 2,
}
}
diff = deepdiff.DeepDiff(dict_1, dict_2)
print(json.dumps(diff, indent=4))
Wynik
{
"values_changed": {
"root['a']": {
"new_value": 2,
"old_value": 1
},
"root['nested']['b']": {
"new_value": 2,
"old_value": 1
}
}
}
Uwaga o ładnym wydrukowaniu wyniku do kontroli: powyższy kod działa, jeśli oba nagrania mają te same klucze atrybutów (z możliwie różnymi wartościami atrybutów jak w przykładzie). Jeśli jednak "extra"
atrybut jest obecny, to jeden ze słowników, json.dumps()
kończy się niepowodzeniem
TypeError: Object of type PrettyOrderedSet is not JSON serializable
Rozwiązanie: użyj diff.to_json()
i json.loads()
/ json.dumps()
do ładnego wydruku:
import deepdiff
import json
dict_1 = {
"a": 1,
"nested": {
"b": 1,
},
"extra": 3
}
dict_2 = {
"a": 2,
"nested": {
"b": 2,
}
}
diff = deepdiff.DeepDiff(dict_1, dict_2)
print(json.dumps(json.loads(diff.to_json()), indent=4))
Wynik:
{
"dictionary_item_removed": [
"root['extra']"
],
"values_changed": {
"root['a']": {
"new_value": 2,
"old_value": 1
},
"root['nested']['b']": {
"new_value": 2,
"old_value": 1
}
}
}
Alternatywnie: użyj pprint
, powoduje inne formatowanie:
import pprint
# same code as above
pprint.pprint(diff, indent=4)
Wynik:
{ 'dictionary_item_removed': [root['extra']],
'values_changed': { "root['a']": { 'new_value': 2,
'old_value': 1},
"root['nested']['b']": { 'new_value': 2,
'old_value': 1}}}
x == y
powinno być prawdziwe zgodnie z stackoverflow.com/a/5635309/186202