Using .format() to format a list with field width arguments

Question:

I recently (finally?) started to use .format() and
have a perhaps a bit obscure question about it.

Given

res = ['Irene Adler', 35,  24.798]

and

(1) print('{0[0]:10s} {0[1]:5d} {0[2]:.2f}'.format(res))
(2) print('{:{}s} {:{}d} {:{}f}'.format(res[0], 10, res[1], 5, res[2], .2))

work great and both print:

Irene Adler    35 24.80
Irene Adler    35 24.80

I didn’t know that I could deal with lists as in (1) which is neat. I
had seen about field width arguments (2) with the old % formatting before.

My question is about wanting to do something like this which combines (1) and (2):

(3) print('{0[0]:{}s} {0[1]:{}d} {0[2]:{}f}'.format(res, 10, 5, .2))

However, I am unable to do this, and I haven’t been able to figure out
from the documentation if this even possible. It would be nice to just
supply the list to be printed, and the arguments for width.

By the way, I also tried this (w/o luck):

args = (10, 5, .2)
(4) print('{0[0]:{}s} {0[1]:{}d} {0[2]:{}f}'.format(res, args))

In both instances I got:

D:UsersblablaDesktop>python tmp.py
Traceback (most recent call last):
  File "tmp.py", line 27, in <module>
    print('{0[0]:{}s} {0[1]:{}d} {0[2]:{}f}'.format(res, 10, 5, .2))
ValueError: cannot switch from manual field specification to automatic field numbering

D:UsersblablaDesktop>python tmp.py
Traceback (most recent call last):
  File "tmp.py", line 35, in <module>
    print('{0[0]:{}s} {0[1]:{}d} {0[2]:{}f}'.format(res, args))
ValueError: cannot switch from manual field specification to automatic field numbering

I also tried using zip() to combine the two sequences without luck.

My question is:

Can I specify a list to be printed effectively doing what I was trying
to unsuccessfully do in (3) and (4) (clearly if this is possible, I’m
not using the right syntax) and if so, how?

Asked By: Levon

||

Answers:

The error message

ValueError: cannot switch from manual field specification to automatic field numbering

pretty much says it all: You need to give explicit field indices everwhere, and

print('{0[0]:{1}s} {0[1]:{2}d} {0[2]:{3}f}'.format(res, 10, 5, .2))

works fine.

Answered By: Sven Marnach

If you want to use .format(res, args), you can specify all indices in the format string like that:

>>> print('{0[0]:{1[0]}s} {0[1]:{1[1]}d} {0[2]:{1[2]}f}'.format(res, args))
Irene Adler    35 24.80

But, if you want to make the format string without indices, you can create a tuple of consecutive pairs (res[0], args[0], ... , res[-1], args[-1]).

This is done by this idiom:

>>> sum(zip(res, args), ())
('Irene Adler', 10, 35, 5, 24.798, 0.2)

You can now pass this into a simplified format string (via *-argument):

>>> fmt = sum(zip(res, args), ())
>>> print('{:{}s} {:{}d} {:{}f}'.format(*fmt))
('Irene Adler', 10, 35, 5, 24.798, 0.2)

This can, of course, be done on one line:

>>> print('{:{}s} {:{}d} {:{}f}'.format(*sum(zip(res, args), ())))
Irene Adler    35 24.80

To make it readable, I would name the transformation:

def flat_pairs(iterable1, iterable2):
    return sum(zip(iterable1, iterable2), ())

# (...)

>>> print('{:{}s} {:{}d} {:{}f}'.format(*flat_pairs(res, args)))
Irene Adler    35 24.80

I think the last one is a reasonable trade-off between readability, convenience, and of course – showing off your Pythonic way of thinking, which is the primary motivation for playing with such stuff.

Answered By: Tomasz Gandor
Categories: questions Tags: , ,
Answers are sorted by their score. The answer accepted by the question owner as the best is marked with
at the top-right corner.