Ответ 1
Это хорошо для вас?
geopoint = {'latitude':41.123,'longitude':71.091}
print('{latitude} {longitude}'.format(**geopoint))
Я большой поклонник использования словарей для форматирования строк. Это помогает мне прочитать строковый формат, который я использую, а также позволить мне использовать существующие словари. Например:
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)
Это хорошо для вас?
geopoint = {'latitude':41.123,'longitude':71.091}
print('{latitude} {longitude}'.format(**geopoint))
Чтобы распаковать словарь в аргументы ключевого слова, используйте **
. Кроме того, поддержка форматирования нового стиля относится к атрибутам объектов и элементов сопоставлений:
'{0[latitude]} {0[longitude]}'.format(geopoint)
'The title is {0.title}s'.format(a) # the a from your first example
Поскольку 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'
Вам просто нужно помнить, чтобы использовать другие символы кавычек во вложенных кавычках. Другим преимуществом этого подхода является то, что он намного быстрее, чем вызов метода форматирования.
print("{latitude} {longitude}".format(**geopoint))
Поскольку вопрос относится к Python 3, здесь используется новый синтаксис f-строки:
>>> geopoint = {'latitude':41.123,'longitude':71.091}
>>> print(f'{geopoint["latitude"]} {geopoint["longitude"]}')
41.123 71.091
Обратите внимание на внешние одинарные кавычки и внутренние двойные кавычки (вы также можете сделать это наоборот).
Синтаксис 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
В большинстве ответов форматируются только значения из 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