Come posso formattare una stringa usando un dizionario in python-3.x?


215

Sono un grande fan dell'uso dei dizionari per formattare le stringhe. Mi aiuta a leggere il formato di stringa che sto usando e mi permette di sfruttare i dizionari esistenti. Per esempio:

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()

Tuttavia, non riesco a capire la sintassi di Python 3.x per fare lo stesso (o se ciò è persino possibile). Vorrei fare quanto segue

# 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)

Risposte:


15

Poiché la domanda è specifica di Python 3, ecco la nuova sintassi f-string , disponibile da Python 3.6:

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

Nota le virgolette singole esterne e le doppie virgolette interne (puoi farlo anche al contrario).


Direi che l'uso della f-string è più allineato all'approccio python3.
Jonatas CD,

2
Tieni presente che le stringhe f sono nuove in Python 3.6 e non in 3.5.
Hugo,

410

Ti va bene?

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

2
Ho provato questo e ha funzionato. Ma non capisco l'uso della "notazione puntatore". So che Python non usa i puntatori, è un esempio di kwargs?
Homunculus Reticulli,

2
@HomunculusReticulli Questo è un parametro di formato (larghezza minima del campo), non un puntatore a uno stile C ++ del puntatore. docs.python.org/release/2.4.4/lib/typesseq-strings.html
D.Rosado

29
Python 3.2 introdotto format_map. Simile a str.format(**mapping), tranne che mappingviene utilizzato direttamente e non copiato in a dict. Ciò è utile se, ad esempio, mappingè una sottoclasse di dict
diapir

1
@eugene Cosa fa ** in un dizionario Python? Non penso che crei un oggetto perché print (** geopoint) non riesce a dare un errore di sintassi
Nityesh Agarwal

4
@NityeshAgarwal diffonde il dizionario con le coppie nome = valore come singoli argomenti, cioè print(**geopoint)è uguale a print(longitude=71.091, latitude=41.123). In molte lingue, è noto come operatore splat . In JavaScript, si chiama operatore di diffusione . In Python non esiste un nome particolare assegnato a questo operatore.
abhisekp,

79

Per decomprimere un dizionario in argomenti di parole chiave, utilizzare **. Inoltre, la formattazione di nuovo stile supporta il riferimento ad attributi di oggetti ed elementi di mappature:

'{0[latitude]} {0[longitude]}'.format(geopoint)
'The title is {0.title}s'.format(a) # the a from your first example

2
Trovo migliore questa risposta, poiché l'aggiunta dell'indice di posizione per il segnaposto rende il codice più esplicito e più facile da usare. Soprattutto se uno ha qualcosa del genere:'{0[latitude]} {1[latitude]} {0[longitude]} {1[longitude]}'.format(geopoint0, geopoint1)
Løiten,

1
Questo è utile se stai usando un defaultdicte non hai tutti i tasti
Whymarrh

65

Dato che Python 3.0 e 3.1 sono EOL e nessuno li usa, puoi e dovresti usare str.format_map(mapping)(Python 3.2+):

Simile a str.format(**mapping), tranne per il fatto che la mappatura viene utilizzata direttamente e non copiata in adict . Ciò è utile se ad esempio il mapping è una dictsottoclasse.

Ciò significa che è possibile utilizzare ad esempio un defaultdictche imposta (e restituisce) un valore predefinito per le chiavi mancanti:

>>> 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'

Anche se la mappatura fornita è una dict, non una sottoclasse, probabilmente sarebbe comunque leggermente più veloce.

Tuttavia, la differenza non è grande

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

poi

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

è circa 10 ns (2%) più veloce di

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

sul mio Python 3.4.3. La differenza sarebbe probabilmente maggiore poiché più chiavi sono nel dizionario e


Si noti che il linguaggio del formato è molto più flessibile di quello però; possono contenere espressioni indicizzate, accessi agli attributi e così via, in modo da poter formattare un intero oggetto o 2 di essi:

>>> 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'

A partire dalla 3.6 puoi usare anche le stringhe interpolate:

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

Devi solo ricordare di usare gli altri caratteri tra virgolette. Un altro aspetto positivo di questo approccio è che è molto più veloce di chiamare un metodo di formattazione.


Bello, ci sono miglioramenti delle prestazioni format? (Dato che non è stato copiato in un dict)
Bhargav Rao

2
@BhargavRao non molto, 2%: D
Antti Haapala

@BhargavRao se stai cercando prestazioni, usa questo '%(latitude)s %(longitude)s'%geopoint;)
Fino

20
print("{latitude} {longitude}".format(**geopoint))

6

La sintassi di Python 2 funziona anche in 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

inoltre è notevolmente più performante di f""o "".format();)
Fino

2
geopoint = {'latitude':41.123,'longitude':71.091}

# working examples.
print(f'{geopoint["latitude"]} {geopoint["longitude"]}') # from above answer
print('{geopoint[latitude]} {geopoint[longitude]}'.format(geopoint=geopoint)) # alternate for format method  (including dict name in string).
print('%(latitude)s %(longitude)s'%geopoint) # thanks @tcll

1
ne hai perso uno;) anche print('%(latitude)s %(longitude)s'%geopoint)questo è significativamente più veloce dell'altro 2
Fino

@tcll In realtà volevo gli esempi, in cui posso usare il nome del dizionario all'interno della stringa. Qualcosa del genere'%(geopoint["latitude"])s %(geopoint["longitude"])s'%{"geopoint":geopoint}
sceicco Abdul Wahid il

1

La maggior parte delle risposte ha formattato solo i valori del dict.

Se vuoi formattare anche la chiave nella stringa puoi usare dict.items () :

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

Produzione:

('latitudine', 41.123) ('longitudine', 71.091)

Se vuoi formattare in modo arbitrario, cioè non mostrare i valori-chiave come le tuple:

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

Produzione:

la latitudine è 41.123 e la longitudine è 71.091


nota che c'è una possibilità che la 'longitudine' possa venire prima della 'latitudine' da geopoint.items();)
Fino
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.