пользовательский порядок сортировки ключей JSON в Python - PullRequest
15 голосов
/ 09 декабря 2010

Есть ли в Python 2.6 какой-нибудь способ предоставить пользовательский ключ или функцию cmp для sort_keys JSON?

У меня есть список диктов, поступающих из JSON, например:

[
  {
    "key": "numberpuzzles1",
    "url": "number-puzzle-i.html",
    "title": "Number Puzzle I",
    "category": "nestedloops",
    "points": "60",
    "n": "087"
  },
  {
     "key": "gettingindividualdigits",
     "url": "getting-individual-digits.html",
     "title": "Getting Individual Digits",
     "category": "nestedloops",
     "points": "80",
     "n": "088"
  }
]

... который я сохранил в списке переменных assigndb.Я хотел бы иметь возможность загружать в JSON, изменять его и сериализовать обратно с dumps (или любым другим), сохраняя при этом порядки ключей без изменений .

До сих пор я пробовал что-то вроде этого:

ordering = {'key': 0, 'url': 1, 'title': 2, 'category': 3,
             'flags': 4, 'points': 5, 'n': 6}

def key_func(k):
    return ordering[k]

# renumber assignments sequentially
for (i, a) in enumerate(assigndb):
    a["n"] = "%03d" % (i+1)

s = json.dumps(assigndb, indent=2, sort_keys=True, key=key_func)

... но, конечно, dumps не поддерживает пользовательский ключ, как list.sort().Может, что-то с кастомом JSONEncoder?Кажется, я не могу этого добиться.

Ответы [ 4 ]

12 голосов
/ 09 декабря 2010

Идея (протестирована с 2.7):

import json
import collections
json.encoder.c_make_encoder = None
d = collections.OrderedDict([("b", 2), ("a", 1)])
json.dumps(d)
# '{"b": 2, "a": 1}'

См .: OrderedDict + Issue 6105 . Кажется, что взломать c_make_encoder нужно только для Python 2.x. Не прямое решение, потому что вы должны изменить dict s на OrderedDict s, но оно все еще может быть использовано. Я проверил библиотеку json ( encode.py ), и заказ был жестко закодирован:

if _sort_keys:
    items = sorted(dct.items(), key=lambda kv: kv[0])
4 голосов
/ 09 декабря 2010

Это некрасиво, но в случае, если решение Tokland не работает для вас:

data = [{'category': 'nestedloops', 'title': 'Number Puzzle I', 'url': 'number-puzzle-i.html', 'n': '087', 'points': '60', 'key': 'numberpuzzles1'}, {'category': 'nestedloops', 'title': 'Getting Individual Digits', 'url': 'getting-individual-digits.html', 'n': '088', 'points': '80', 'key': 'gettingindividualdigits'}]
ordering = {'key': 0, 'url': 1, 'title': 2, 'category': 3,
            'flags': 4, 'points': 5, 'n': 6}
outlist = []
for d in data:
    outlist.append([])
    for k in sorted(d.keys(), key=lambda k: ordering[k]):
        outlist[-1].append(json.dumps({k: d[k]}))

for i, l in enumerate(outlist):
    outlist[i] = "{" + ",".join((s[1:-1] for s in outlist[i])) + "}"

s = "[" + ",".join(outlist) + "]"
0 голосов
/ 25 октября 2016

У меня была такая же проблема, и collections.OrderedDict просто не подходил для этой задачи, потому что все было упорядочено по алфавиту Поэтому я написал нечто похожее на решение Эндрю Кларка:

def json_dumps_sorted(data, **kwargs):
    sorted_keys = kwargs.get('sorted_keys', tuple())
    if not sorted_keys:
        return json.dumps(data)
    else:
        out_list = []
        for element in data:
            element_list = []
            for key in sorted_keys:
                if key in element:
                    element_list.append(json.dumps({key: element[key]}))
            out_list.append('{{{}}}'.format(','.join((s[1:-1] for s in element_list))))
        return '[{}]'.format(','.join(out_list))

Вы используете это так:

json_string = json_dumps_sorted([
    {
        "key": "numberpuzzles1",
        "url": "number-puzzle-i.html",
        "title": "Number Puzzle I",
        "category": "nestedloops",
        "points": "60",
        "n": "087"
    }, {
        "key": "gettingindividualdigits",
        "url": "getting-individual-digits.html",
        "title": "Getting Individual Digits",
        "category": "nestedloops",
        "points": "80",
        "n": "088"
    }
], sorted_keys=(
    'key', 
    'url', 
    'title', 
    'category',
    'flags', 
    'points', 
    'n'
))
0 голосов
/ 04 декабря 2015

Спасибо. Мне нужно было поместить ключ метки времени: значение в верхней части моего объекта JSON, несмотря ни на что. Очевидно, сортировка ключей испортила это, поскольку это начинается с "t".

Используя что-то вроде этого, сразу же добавив ключ метки времени в dict_data, сработало:

d = collections.OrderedDict(dict_data)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...