Использование Python's Format Specification Mini-Language для выравнивания поплавков - PullRequest
16 голосов
/ 03 марта 2012

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

Например, скажем, у меня есть три следующих списка:

job_IDs = ['13453', '123', '563456'];
memory_used = [30, 150.54, 20.6];
memory_units = ['MB', 'GB', 'MB'];

Я бы хотел пройтись по этим трем спискам и вывести

Job 13453:   30      MB
Job 123:    150.54   MB
Job 563456:  20.6    GB

Пока я пробовал:

for i in range(len(jobIDs)):
   my_str = "{item:15}{value:6} {units:3}".format( 
   item='Job ' + job_IDs[i] + ':' , value=str(memories[i]),units=memory_units[i]);

   print my_str

который печатает:

Job 13453:   30      MB
Job 123:     150.54  MB
Job 563456:  20.6    GB

, что почти правильно, но оно не выравнивает числа вокруг десятичной запятой. Как я могу использовать Python Mini Format Language со спецификацией формата , чтобы сделать это так, как мне нужно?

Ответы [ 4 ]

11 голосов
/ 03 марта 2012

Это то, что вы хотите:

for i in range(len(job_IDs)):
    print "Job {item:15} {value[0]:>6}.{value[1]:<6} {units:3}".format(item=job_IDs[i]+':', value=memory_used[i].split('.') if '.' in memory_used[i] else (memory_used[i], '0'), units=memory_units[i])

Вот как это работает:

Это основная часть: value=memory_used[i].split('.') if '.' in memory_used[i] else (memory_used[i], '0'), что означает: если есть десятичная точка, разделите строку на целую и десятичную часть или установите для десятичной части значение 0.

Затем в строке формата: {value[0]:>6}.{value[1]:<6} означает, что вся часть сместилась вправо, затем точка, затем десятичная часть сместилась влево.

который печатает:

Job 13453:              30.0      MB
Job 123:               150.54     GB
Job 563456:             20.6      MB
10 голосов
/ 03 марта 2012

Вот еще одна реализация, основанная на идее .split('.'). Это может быть более читабельным. Разделить на '.', выровнять по правому краю левую часть, выровнять по левому краю правую часть:

width = max(map(len, job_IDs)) # width of "job id" field 
for jid, mem, unit in zip(job_IDs, memory_used, memory_units):
  print("Job {jid:{width}}: {part[0]:>3}{part[1]:1}{part[2]:<3} {unit:3}".format(
    jid=jid, width=width, part=str(mem).partition('.'), unit=unit))

выход

Job 13453 :  30     MB 
Job 123   : 150.54  GB 
Job 563456:  20.6   MB 
3 голосов
/ 14 июня 2012

В случае, если это поможет, вот аналогичная функция, которую я использую:

def align_decimal(number, left_pad=7, precision=2):
    """Format a number in a way that will align decimal points."""
    outer = '{0:>%i}.{1:<%i}' % (left_pad, precision)
    inner = '{:.%if}' % (precision,)
    return outer.format(*(inner.format(number).split('.')))

Это позволяет фиксированной точности после десятичной точки.

1 голос
/ 03 марта 2012

Это делает это:

import re

job_IDs = ['13453', '123', '563456']
memory_used = ['30', '150.54', '20.6']
memory_units = ['MB', 'GB', 'MB']

for i in range(len(job_IDs)):
    lh=re.match(r'(\d+)',memory_used[i]).group(1)
    if '.' in memory_used[i]:
        rh=re.match(r'(\d+)\.(\d+)',memory_used[i]).group(2)
        sep='.'
    else:
        rh=''    
        sep=' '


    my_str = "{item:15}{l:>6}{s:1}{r:6} {units:3}".format( 
    item='Job ' + job_IDs[i] + ':' , l=lh,s=sep,r=rh,units=memory_units[i])

    print my_str

Чтобы получить точный результат, вам нужно разбить строки на необязательный '.'

Вы также можете конвертировать эти строки в плавающие.

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