Как напечатать строку фиксированной ширины?

105

У меня есть этот код (распечатываю вхождение всех перестановок в строке)

def splitter(str):
    for i in range(1, len(str)):
        start = str[0:i]
        end = str[i:]
        yield (start, end)
        for split in splitter(end):
            result = [start]
            result.extend(split)
            yield result    

el =[];

string = "abcd"
for b in splitter("abcd"):
    el.extend(b);

unique =  sorted(set(el));

for prefix in unique:
    if prefix != "":
        print "value  " , prefix  , "- num of occurrences =   " , string.count(str(prefix));

Я хочу напечатать все вхождения перестановок в строковой переменной.

так как перестановка не той же длины, я хочу исправить ширину и распечатать ее в красивом, не похожем на этот:

value   a - num of occurrences =    1
value   ab - num of occurrences =    1
value   abc - num of occurrences =    1
value   b - num of occurrences =    1
value   bc - num of occurrences =    1
value   bcd - num of occurrences =    1
value   c - num of occurrences =    1
value   cd - num of occurrences =    1
value   d - num of occurrences =    1

Как я formatмогу это сделать?

Я нашел эти сообщения, но они не подходили для буквенно-цифровых строк:

Форматирование строки Python с фиксированной шириной

Установка фиксированной длины с помощью Python

0x90
источник
1
как насчет print '% 10s'% 'mystring'
TJD
2
Удивлен, что "\t"ни в одном решении не указан вариант.

Ответы:

115

ИЗМЕНИТЬ 2013-12-11 - Этот ответ очень старый. Он по-прежнему действителен и правильный, но люди, которые смотрят на него, должны предпочесть новый синтаксис формата .

Вы можете использовать форматирование строк следующим образом:

>>> print '%5s' % 'aa'
   aa
>>> print '%5s' % 'aaa'
  aaa
>>> print '%5s' % 'aaaa'
 aaaa
>>> print '%5s' % 'aaaaa'
aaaaa

В принципе:

  • как %сообщает характер Python он должен будет заменитель что - то знак
  • в sсимвольных сообщает питон маркера будет строка
  • 5(или любой номер , который вы хотите) сообщает питон раздуть строка с пробелами до 5 символов.

В вашем конкретном случае возможная реализация может выглядеть так:

>>> dict_ = {'a': 1, 'ab': 1, 'abc': 1}
>>> for item in dict_.items():
...     print 'value %3s - num of occurances = %d' % item # %d is the token of integers
... 
value   a - num of occurances = 1
value  ab - num of occurances = 1
value abc - num of occurances = 1

ПРИМЕЧАНИЕ: Просто интересно, знаете ли вы о существовании itertoolsмодуля . Например, вы можете получить список всех ваших комбинаций в одной строке с помощью:

>>> [''.join(perm) for i in range(1, len(s)) for perm in it.permutations(s, i)]
['a', 'b', 'c', 'd', 'ab', 'ac', 'ad', 'ba', 'bc', 'bd', 'ca', 'cb', 'cd', 'da', 'db', 'dc', 'abc', 'abd', 'acb', 'acd', 'adb', 'adc', 'bac', 'bad', 'bca', 'bcd', 'bda', 'bdc', 'cab', 'cad', 'cba', 'cbd', 'cda', 'cdb', 'dab', 'dac', 'dba', 'dbc', 'dca', 'dcb']

и вы можете получить количество вхождений, используя combinationsвместе с count().

Mac
источник
24
Возможно, вам стоит упомянуть, что отрицательные числа дают выровненный по левому краю вывод с заполнением; это вряд ли интуитивно понятно для новичка.
Tripleee
+1 за @tripleee, если бы ваши отрицательные числа не давали выровненный по левому краю комментарий, я бы бился головой дольше ... thx m8.
Briford Wylie
Это гораздо более интуитивно понятно и лаконично, чем новый str.format. Я не понимаю, почему в
питоне
Есть ли способ заполнить пробелы определенным символом? Например, если нам нужно вывести «05» вместо «5»
Харшит Джиндал
1
Вот еще несколько приемов для элегантной печати фиксированной ширины с f-строками на Medium .
pfabri
200

Я считаю использование str.formatболее элегантным:

>>> '{0: <5}'.format('ss')
'ss   '
>>> '{0: <5}'.format('sss')
'sss  '
>>> '{0: <5}'.format('ssss')
'ssss '
>>> '{0: <5}'.format('sssss')
'sssss'

Если вы хотите выровнять строку по правому краю, используйте >вместо <:

>>> '{0: >5}'.format('ss')
'   ss'

Изменить: как упоминалось в комментариях: 0 указывает переданный индекс аргумента str.format().

0x90
источник
5
Кроме того, 0 указывает позицию аргумента формата, поэтому вы можете делать еще две вещи: '{<5}'.format('ss') 'ss 'точно так же, как раньше, но без 0, делает то же самое, или 'Second {1: <5} and first {0: <5}'.format('ss', 'sss') 'Second sss and first ss 'вы можете переупорядочить или даже вывести одну и ту же переменную много раз в один вывод строка.
mightypile
14
Я больше не могу редактировать предыдущий комментарий, который в нем нуждается. {<5}не работает, но {: <5}работает без значения индекса.
mightypile
9
Вот мини-язык спецификации формата Python, в котором описаны эти строки формата и дополнительные параметры. Для быстрой справки, пространство в {0: <5}- это [fill] , <есть [align]и 5есть[width]
cod3monk3y
2
Эта 5 может быть заменой переменной>>> print width 20 >>> print "{0: <{width}}".format("ssssss", width=width).split('\n') ['ssssss '] >>>
user2763554 04
4
Вы также можете использовать числа и просто перечислить переменные по порядку width=10; "{0: <{1}}".format('sss', width). Или даже не '{: <{}}'.format('sss', width)
указывайте
74

Первоначально опубликовано как редактирование ответа @ 0x90, но оно было отклонено за отклонение от первоначального намерения публикации и рекомендовано для публикации в качестве комментария или ответа, поэтому я включаю здесь короткую рецензию.

В дополнение к ответу от @ 0x90, синтаксис можно сделать более гибким, используя переменную ширины (согласно комментарию @ user2763554):

width=10
'{0: <{width}}'.format('sss', width=width)

Кроме того, вы можете сделать это выражение короче, используя только числа и полагаясь на порядок аргументов, переданных в format:

width=10
'{0: <{1}}'.format('sss', width)

Или даже оставьте все числа для максимальной, потенциально не подразумеваемой питоном, компактности:

width=10
'{: <{}}'.format('sss', width)

Обновление 2017-05-26

С введением в Python 3.6 форматированных строковых литералов (сокращенно «f-strings») теперь можно получить доступ к ранее определенным переменным с более коротким синтаксисом:

>>> name = "Fred"
>>> f"He said his name is {name}."
'He said his name is Fred.'

Это также относится к форматированию строк

>>> width=10
>>> string = 'sss'
>>> f'{string: <{width}}'
'sss       '
Joelostblom
источник
Мне очень нравится этот ответ!
Syrtis Major
10

formatэто определенно самый элегантный способ, но, черт возьми, вы не можете использовать его с loggingмодулем python , поэтому вот как вы можете сделать это с помощью %форматирования:

formatter = logging.Formatter(
    fmt='%(asctime)s | %(name)-20s | %(levelname)-10s | %(message)s',
)

Здесь значок -указывает на выравнивание по левому краю, а цифра перед ним sуказывает на фиксированную ширину.

Пример вывода:

2017-03-14 14:43:42,581 | this-app             | INFO       | running main
2017-03-14 14:43:42,581 | this-app.aux         | DEBUG      | 5 is an int!
2017-03-14 14:43:42,581 | this-app.aux         | INFO       | hello
2017-03-14 14:43:42,581 | this-app             | ERROR      | failed running main

Дополнительная информация в документах здесь: https://docs.python.org/2/library/stdtypes.html#string-formatting-operations

Райантак
источник
Однако это не приведет к сокращению строк длиной более 20 символов. Используйте, '%(name)20.20s'который устанавливает 20 как минимальную, так и максимальную длину строки!
xjcl