Как мне отформатировать строку, используя словарь в python-3.x? - PullRequest
177 голосов
/ 10 мая 2011

Я большой поклонник использования словарей для форматирования строк. Это помогает мне читать формат строки, который я использую, а также позволяет использовать существующие словари. Например:

class MyClass:
    def __init__(self):
        self.title = 'Title'

a = MyClass()
print 'The title is %(title)s' % a.__dict__

path = '/path/to/a/file'
print 'You put your file here: %(path)s' % locals()

Однако я не могу понять синтаксис Python 3.x для того же (или, если это вообще возможно). Я хотел бы сделать следующее

# Fails, KeyError 'latitude'
geopoint = {'latitude':41.123,'longitude':71.091}
print '{latitude} {longitude}'.format(geopoint)

# Succeeds
print '{latitude} {longitude}'.format(latitude=41.123,longitude=71.091)

Ответы [ 7 ]

343 голосов
/ 10 мая 2011

Это хорошо для тебя?

geopoint = {'latitude':41.123,'longitude':71.091}
print('{latitude} {longitude}'.format(**geopoint))
74 голосов
/ 10 мая 2011

Чтобы распаковать словарь в аргументы ключевого слова, используйте **. Кроме того, форматирование нового стиля поддерживает ссылки на атрибуты объектов и элементов отображений:

'{0[latitude]} {0[longitude]}'.format(geopoint)
'The title is {0.title}s'.format(a) # the a from your first example
50 голосов
/ 14 апреля 2016

Поскольку Python 3.0 и 3.1 являются EOL'ами, и никто не использует их, вы можете и должны использовать str.format_map(mapping) (Python 3.2 +):

По аналогии с str.format(**mapping), за исключением того, что сопоставление используется напрямую и не копируется в dict.Это полезно, если, например, отображение является подклассом dict.

Это означает, что вы можете использовать, например, defaultdict, который установит (и вернет) значение по умолчанию для ключей, которыеотсутствуют:

>>> from collections import defaultdict
>>> vals = defaultdict(lambda: '<unset>', {'bar': 'baz'})
>>> 'foo is {foo} and bar is {bar}'.format_map(vals)
'foo is <unset> and bar is baz'

Даже если предоставленное отображение является dict, а не подклассом, это, вероятно, все равно будет немного быстрее.

Хотя разница невелика, учитывая

>>> d = dict(foo='x', bar='y', baz='z')

, тогда

>>> 'foo is {foo}, bar is {bar} and baz is {baz}'.format_map(d)

примерно на 10 нс (2%) быстрее, чем

>>> 'foo is {foo}, bar is {bar} and baz is {baz}'.format(**d)

на моем Python 3.4.3.Разница, вероятно, будет больше, поскольку в словаре будет больше ключей, и


Обратите внимание, что язык форматирования гораздо более гибкий, чем этот;они могут содержать индексированные выражения, доступ к атрибутам и т. д., поэтому вы можете отформатировать целый объект или 2 из них:

>>> p1 = {'latitude':41.123,'longitude':71.091}
>>> p2 = {'latitude':56.456,'longitude':23.456}
>>> '{0[latitude]} {0[longitude]} - {1[latitude]} {1[longitude]}'.format(p1, p2)
'41.123 71.091 - 56.456 23.456'

Начиная с версии 3.6 вы можете использоватьИнтерполированные строки тоже:

>>> f'lat:{p1["latitude"]} lng:{p1["longitude"]}'
'lat:41.123 lng:71.091'

Вам просто нужно помнить, чтобы использовать другие символы кавычек во вложенных кавычках.Другим преимуществом этого подхода является то, что он намного быстрее, чем вызывает метод форматирования.

15 голосов
/ 10 мая 2011
print("{latitude} {longitude}".format(**geopoint))
6 голосов
/ 28 сентября 2017

Поскольку вопрос относится только к Python 3, здесь используется новый синтаксис f-строки :

>>> geopoint = {'latitude':41.123,'longitude':71.091}
>>> print(f'{geopoint["latitude"]} {geopoint["longitude"]}')
41.123 71.091

Обратите внимание на внешние одинарные кавычки и внутренние двойные кавычки (вы также можете сделать это наоборот).

5 голосов
/ 11 мая 2011

Синтаксис Python 2 работает и в Python 3:

>>> class MyClass:
...     def __init__(self):
...         self.title = 'Title'
... 
>>> a = MyClass()
>>> print('The title is %(title)s' % a.__dict__)
The title is Title
>>> 
>>> path = '/path/to/a/file'
>>> print('You put your file here: %(path)s' % locals())
You put your file here: /path/to/a/file
0 голосов
/ 10 мая 2019

В большинстве ответов форматируются только значения из dict.

Если вы хотите , также отформатируйте ключ в строку, которую вы можете использовать dict.items () :

geopoint = {'latitude':41.123,'longitude':71.091}
print("{} {}".format(*geopoint.items()))

Выход:

(«широта», 41.123) («долгота», 71.091)

Если вы хотите выполнить произвольное форматирование, то есть не показывать значения ключей, такие как кортежи:

from functools import reduce
print("{} is {} and {} is {}".format(*reduce((lambda x, y: x + y), [list(item) for item in geopoint.items()])))

Выход:

широта - 41.123, а долгота - 71.091

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