Як надрукувати словник за рядком у Python?


166

Це словник

cars = {'A':{'speed':70,
        'color':2},
        'B':{'speed':60,
        'color':3}}

Використовуючи це for loop

for keys,values in cars.items():
    print(keys)
    print(values)

Він друкує наступне:

B
{'color': 3, 'speed': 60}
A
{'color': 2, 'speed': 70}

Але я хочу, щоб програма надрукувала його так:

B
color : 3
speed : 60
A
color : 2
speed : 70

Я тільки почав вивчати словники, тому не знаю, як це зробити.

Відповіді:


142
for x in cars:
    print (x)
    for y in cars[x]:
        print (y,':',cars[x][y])

вихід:

A
color : 2
speed : 70
B
color : 3
speed : 60

12
Я знаю, що це старе, але я подумав, що варто згадати, що це не працює, якщо автомобілі [x] мають цілі числа. Це не те, про що вимагала ОП, тому я просто говорю це будь-кому, хто натрапляє на це, припускаючи, що це бланкетне рішення.
Даррель Холт

@DarrelHolt Ви знаєте, як змусити його працювати з цілими числами? Тому що це проблема, з якою я
стикаюсь в

@theprowler Найближче до вирішення проблеми я можу дійти, якщо cars = {1:4, 2:5}тоді cars[x]це ціле число, відображене на ключ, xа не набір, відображений на ключ x. У цьому випадку вам не потрібно використовувати for y in cars[x]:рядок, оскільки ви отримуєте лише одне значення, якщо ви не використовуєте щось на зразок списку чи набору цілих чисел, то воно повинно працювати. Вибачте, пройшло кілька місяців, тому я не можу повністю пригадати, як я прийшов до висновку свого попереднього коментаря. Ви можете надіслати мені свій код, і я можу побачити, чи допоможу я.
Даррель Холт

Хм. Я думаю, моя проблема ще гірша за це. В основному я розібрав деякі дані з таблиці HTML, і мені трапилось зберегти їх у словнику, і тепер я намагаюся взяти ці словникові дані і вкласти їх у DataFrame, перш ніж експортувати їх у таблицю Oracle. ... я знаю досить глибоко, але крок, який мене зараз затримує, - це введення даних у DataFrame .... у моєму словнику чомусь є одна клавіша, і всі дані значуться, тому важко намагаючись розмістити його акуратно в рядки та стовпці ..
theprowler

118

Ви можете використовувати jsonмодуль для цього. dumpsФункція в цьому модулі перетворює об'єкт JSON в правильно відформатованої рядки , які потім можна роздрукувати.

import json

cars = {'A':{'speed':70, 'color':2},
        'B':{'speed':60, 'color':3}}

print(json.dumps(cars, indent = 4))

Вихід виглядає так

{
    "А": {
        "колір": 2,
        "швидкість": 70
    },
    "B": {
        "колір": 3,
        "швидкість": 60
    }
}

Документація також визначає купу корисних опцій для цього методу.


2
правда, вміст диктату повинен бути серіалізаційним в json, однак, поданий тут вихід набагато чистіший (наприклад, для людського читання), ніж вихід, отриманий pprint.PrettyPrinter. зокрема в області послідовного відступу та відкидання рядкових префіксів, таких як u'foo.
Буффало Рабор

Я це роблю, print(json.dumps(cars, indent=4, ensure_ascii=False))бо в іншому випадку символи, що не належать до ASCII, не читаються.
Борис

85

Більш узагальненим рішенням, яке обробляє довільно глибоко вкладені дикти та списки, було б:

def dumpclean(obj):
    if isinstance(obj, dict):
        for k, v in obj.items():
            if hasattr(v, '__iter__'):
                print k
                dumpclean(v)
            else:
                print '%s : %s' % (k, v)
    elif isinstance(obj, list):
        for v in obj:
            if hasattr(v, '__iter__'):
                dumpclean(v)
            else:
                print v
    else:
        print obj

Це дає результат:

A
color : 2
speed : 70
B
color : 3
speed : 60

Я зіткнувся з подібною потребою і розвинув більш міцну функцію як вправу для себе. Я включаю його сюди, якщо він може бути корисним для іншого. Під час запуску нос-тесту я також вважаю корисним можливість вказати вихідний потік у виклику, щоб замість нього можна було використовувати sys.stderr.

import sys

def dump(obj, nested_level=0, output=sys.stdout):
    spacing = '   '
    if isinstance(obj, dict):
        print >> output, '%s{' % ((nested_level) * spacing)
        for k, v in obj.items():
            if hasattr(v, '__iter__'):
                print >> output, '%s%s:' % ((nested_level + 1) * spacing, k)
                dump(v, nested_level + 1, output)
            else:
                print >> output, '%s%s: %s' % ((nested_level + 1) * spacing, k, v)
        print >> output, '%s}' % (nested_level * spacing)
    elif isinstance(obj, list):
        print >> output, '%s[' % ((nested_level) * spacing)
        for v in obj:
            if hasattr(v, '__iter__'):
                dump(v, nested_level + 1, output)
            else:
                print >> output, '%s%s' % ((nested_level + 1) * spacing, v)
        print >> output, '%s]' % ((nested_level) * spacing)
    else:
        print >> output, '%s%s' % (nested_level * spacing, obj)

За допомогою цієї функції вихід OP виглядає приблизно так:

{
   A:
   {
      color: 2
      speed: 70
   }
   B:
   {
      color: 3
      speed: 60
   }
}

що я особисто вважав більш корисним та описовим.

З огляду на дещо менш тривіальний приклад:

{"test": [{1:3}], "test2":[(1,2),(3,4)],"test3": {(1,2):['abc', 'def', 'ghi'],(4,5):'def'}}

Запрошене рішення ОП дає таке:

test
1 : 3
test3
(1, 2)
abc
def
ghi
(4, 5) : def
test2
(1, 2)
(3, 4)

тоді як "покращена" версія дає це:

{
   test:
   [
      {
         1: 3
      }
   ]
   test3:
   {
      (1, 2):
      [
         abc
         def
         ghi
      ]
      (4, 5): def
   }
   test2:
   [
      (1, 2)
      (3, 4)
   ]
}

Я сподіваюся, що це надасть певної цінності наступній людині, яка шукає цей тип функціональності.


11
І якщо формат не надто суворий, можна також використовувати "print json.dumps (obj, indent = 3)". Це дає розумне уявлення більшості структур, хоча воно задихається (в моєму оточенні) на моєму нетривіальному прикладі завдяки використанню кортежу як ключового ...
MrWonderful

7
Чому б просто не використовувати pprint.pprint()тут тоді?
Martijn Pieters

1
майже зробив творця JSON, ні?
користувач2007447

30

У вас є вкладена структура, тому вам також потрібно відформатувати вкладений словник:

for key, car in cars.items():
    print(key)
    for attribute, value in car.items():
        print('{} : {}'.format(attribute, value))

Це відбитки:

A
color : 2
speed : 70
B
color : 3
speed : 60

28

pprint.pprint() є хорошим інструментом для цієї роботи:

>>> import pprint
>>> cars = {'A':{'speed':70,
...         'color':2},
...         'B':{'speed':60,
...         'color':3}}
>>> pprint.pprint(cars, width=1)
{'A': {'color': 2,
       'speed': 70},
 'B': {'color': 3,
       'speed': 60}}



4

Перевірте наступний одношаровий:

print('\n'.join("%s\n%s" % (key1,('\n'.join("%s : %r" % (key2,val2) for (key2,val2) in val1.items()))) for (key1,val1) in cars.items()))

Вихід:

A
speed : 70
color : 2
B
speed : 60
color : 3

Хороший, але я намагався його перетворити, щоб це використовувати sys.modules, але мені не вдалося. Хочете спробувати?
not2qubit

4

Я віддаю перевагу чистому форматуванню yaml:

import yaml
yaml.dump(cars)

вихід:

A:
  color: 2
  speed: 70
B:
  color: 3
  speed: 60

Ви повинні pip install PyYAMLспочатку.
Борис

0
###newbie exact answer desired (Python v3):
###=================================
"""
cars = {'A':{'speed':70,
        'color':2},
        'B':{'speed':60,
        'color':3}}
"""

for keys, values in  reversed(sorted(cars.items())):
    print(keys)
    for keys,values in sorted(values.items()):
        print(keys," : ", values)

"""
Output:
B
color  :  3
speed  :  60
A
color  :  2
speed  :  70

##[Finished in 0.073s]
"""

0
# Declare and Initialize Map
map = {}

map ["New"] = 1
map ["to"] = 1
map ["Python"] = 5
map ["or"] = 2

# Print Statement
for i in map:
  print ("", i, ":", map[i])

#  New : 1
#  to : 1
#  Python : 5
#  or : 2

0

Ось моє рішення проблеми. Я думаю, що це схоже за підходом, але трохи простіше, ніж деякі інші відповіді. Він також дозволяє довільну кількість під словників і, здається, працює для будь-якого типу даних (я навіть тестував його на словнику, який мав функції як значення):

def pprint(web, level):
    for k,v in web.items():
        if isinstance(v, dict):
            print('\t'*level, f'{k}: ')
            level += 1
            pprint(v, level)
            level -= 1
        else:
            print('\t'*level, k, ": ", v)

-1

Зміна коду MrWonderful

import sys

def print_dictionary(obj, ident):
    if type(obj) == dict:
        for k, v in obj.items():
            sys.stdout.write(ident)
            if hasattr(v, '__iter__'):
                print k
                print_dictionary(v, ident + '  ')
            else:
                print '%s : %s' % (k, v)
    elif type(obj) == list:
        for v in obj:
            sys.stdout.write(ident)
            if hasattr(v, '__iter__'):
                print_dictionary(v, ident + '  ')
            else:
                print v
    else:
        print obj

1
Що ви змінили? Який вихід?
Андреас Хафербург
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.