Python 語(yǔ)法淺析:字符串格式化

前序

There should be one - and preferably only one - obvious way to do it.

———— the Zen of Python

意譯:Python提倡用一種悠砚,而且最好是只有一種方法來(lái)完成一件事

雖然 Python 有以上的提倡往堡,但卻在字符串格式化方面,沒有做到這一點(diǎn)窘奏。

字符串格式化

敲黑板厅篓,劃重點(diǎn):在 Python 中有至少三種常見方式實(shí)現(xiàn)字符串格式化:

  1. %-formatting 格式(Python2.6以前酣栈,推薦輸出時(shí)使用)
  2. str.format() 格式(Python2.6惰说,推薦字符串拼接使用)
  3. f-string 格式(Python3.6,推薦使用)

1岳服、printf 風(fēng)格的字符串格式化(%-formatting 格式)

Python2.6 之前只有這一種方式剂公,使用與 C 中 printf 函數(shù)一樣的語(yǔ)法。

基礎(chǔ)語(yǔ)法:format % value(其中 format 為一個(gè)字符串)吊宋,在 format 中的 % 轉(zhuǎn)換標(biāo)記符將被替換為零個(gè)或者多個(gè) value 條目

基礎(chǔ)用法

# coding in Python3.7
print('this is %s blog written in %d%%%02d%%%d %02d:%d:%06.2f' 
      % ('TinyDolphin', 2019, 5, 31, 9, 30, 22.3333))
# this is TinyDolphin blog written in 2019%05%31 09:30:022.33

print('title=%(title)s, author=%(name)s' 
      % {'name': 'TinyDolphin', 'title': 'Python 語(yǔ)法淺析:字符串格式化'})
# title=Python 語(yǔ)法淺析:字符串格式化, author=TinyDolphin

print('%0*.*f' % (6, 2, 2.345))
# 002.35

printf 語(yǔ)法

針對(duì)這種格式化操作纲辽,重要知識(shí)點(diǎn)如下:
1、針對(duì)這些轉(zhuǎn)換標(biāo)記字符贫母,必須按照以下順序:
    % --> (name) --> '#'文兑、'-'盒刚、'0'腺劣、'+'、' ' --> m.n | m --> d因块、s橘原、r、f

2涡上、常用轉(zhuǎn)換類型:
    %s 格式化字符串(str()函數(shù))
    %r 格式化字符串(repr()函數(shù))
    %d 格式化整數(shù)
    %f 格式化浮點(diǎn)數(shù)字郁季,可指定小數(shù)點(diǎn)后的精度

3丁频、常用的轉(zhuǎn)換標(biāo)記字符:
    - 用做左對(duì)齊
    + 在正數(shù)前面顯示加號(hào)( + )
    # 在八進(jìn)制數(shù)前面顯示零('0'),在十六進(jìn)制前面顯示'0x'或者'0X'(取決于用的是'x'還是'X')
    0 顯示的數(shù)字前面填充'0',而不是默認(rèn)的空格
    % '%%'輸出一個(gè)單一的'%'
    m.n 表示 m 是顯示的最小總寬度,n 是小數(shù)點(diǎn)后的位數(shù)
    * 定義寬度或者小數(shù)點(diǎn)精度(用在不能預(yù)先指定m.n的值)
    (var) 映射變量(字典參數(shù))

2、字符串的方式(str.format()格式)

Python2.6 開始答憔,新增了一種格式化字符串的函數(shù) str.format(),它增強(qiáng)了字符串格式化的功能拿霉,比如:支持位置映射筋搏、關(guān)鍵字映射對(duì)象屬性映射糖权、下標(biāo)映射等多種方式

基本語(yǔ)法是通過 {} 和 : 來(lái)代替以前的 %堵腹,例如:'%03.2f'被改寫成 '{:03.2f}'。

兩個(gè)格式化方法:

str.format(*args, **kwargs)
    字符串的格式化操作星澳。
    str 包含字符串字面值 AND {} 括起來(lái)的替換域疚顷。
    每個(gè)替換域:位置參數(shù)的索引 OR 關(guān)鍵字參數(shù)的名稱
    
str.format_map(mapping)
    類似于 str.foramt(**mapping)
    不同之處:mapping 會(huì)被直接使用而不是復(fù)制到一個(gè)dict。

PS:Python 存在內(nèi)置函數(shù) format(value, format_spec):會(huì)轉(zhuǎn)換成 type(value).__format__(value, format_spec)

基礎(chǔ)用法

1、按位置訪問參數(shù)

'{0}-{1}-{2}'.format('a', 'b', 'c')     # 'a-b-c'
'{}-{}-{}'.format('a', 'b', 'c')        # 'a-b-c'
'{2}-{1}-{0}'.format('a', 'b', 'c')     # 'c-b-a'
'{2}-{1}-{0}'.format(*'abc')            # 'c-b-a'

args = ['a', 'b', 'c']
'{2}-{1}-{0}'.format(*args)             # 'c-b-a'

'{0}-{1}-{0}'.format(*['abc', 'def'])   # 'abc-def-abc'

2腿堤、按名稱訪問參數(shù)

'{a}-阀坏'.format(a='1', b='2')          # '1-2'

kwargs = {'a':'1', 'b':'2'}
'{a}-'.format(**kwargs)              # '1-2'

3释液、訪問參數(shù)的屬性

class Point:
    def __init__(self, x, y):
        self.x, self.y = x, y
    def __str__(self):
        return 'Point({self.x}, {self.y})'.format(self=self)
        
str(Point(3, 4))                        # 'Point(3, 4)'

4全释、訪問參數(shù)的項(xiàng)

point = (3, 5)
'X:{0[0]} Y:{0[1]}'.format(point)       # 'X:3 Y:5'

5、替代 %s 和 %r (!s误债、!r)

'str() = {!s}; repr() = {!r}'.format('a', 'a')  # "str() = a; repr() = 'a'"

6浸船、對(duì)齊文本以及制定寬度(:<、:^寝蹈、:>)

'{:<20}'.format('left aligned')         # 'left aligned        '
'{:>20}'.format('right aligned')        # '       right aligned'
'{:^20}'.format('centered')             # '      centered      '
# 使用 '*' 作為填充字符
'{:*^20}'.format('centered')            # '******centered******'

7李命、替代 %+f、%-f 和 %f 以及指定正負(fù)號(hào)(:+箫老、:-封字、:)

'{:+f} {:+f}'.format(3.14, -3.14)       # '+3.140000 -3.140000'
'{: f} {: f}'.format(3.14, -3.14)       # ' 3.140000 -3.140000' 正號(hào)—>空格
'{:-f} {:-f}'.format(3.14, -3.14)       # '3.140000 -3.140000' == '{:f} {:f}'

8、替換 %x 和 %o 以及轉(zhuǎn)換基于不同進(jìn)位制的值(:x耍鬓、:d阔籽、:o、:b)

'int:{0:d} hex:{0:x} oct:{0:o} bin:{0:b}'.format(42)
# 'int:42 hex:2a oct:52 bin:101010'
'int:{0:d} hex:{0:#x} oct:{0:#o} bin:{0:#b}'.format(42)
# 'int:42 hex:0x2a oct:0o52 bin:0b101010'
# '#' : 加上前綴:0x牲蜀、0o笆制、0b

9、使用逗號(hào)作為千位分隔符(:,)

'{:,}'.format(123456789)                # '123,456,789'

10涣达、表示為百分?jǐn)?shù)(:.2%)

'{:.2%}'.format(19/22)                  # '86.36%'

11在辆、使用特定類型的專屬格式化

import datetime
d = datetime.datetime(2019, 6, 10, 22, 5, 13)
'{:%Y-%m-%d %H:%M:%S}'.format(d)
# '2019-06-10 22:05:13'

12、嵌套參數(shù)以及更復(fù)雜的示例

3度苔、格式化字符串字面值(f-string 格式)

是帶有 'f' 或 'F' 前綴的字符串字面值匆篓。

花括號(hào)以外的部分按其字面值處理,除了雙重花括號(hào) '{{' 或 '}}' 會(huì)被替換為相應(yīng)的單個(gè)花括號(hào)寇窑。

語(yǔ)法如下:

單個(gè)花括號(hào) '{'鸦概,標(biāo)示一個(gè)替換字段(以 Python 表達(dá)式打頭)
 + 可能有一個(gè)以嘆號(hào) '!' 標(biāo)示的轉(zhuǎn)換字符 
 + 可能有一個(gè)以冒號(hào) ':' 標(biāo)示的格式說(shuō)明符 
 + 以一個(gè)右花括號(hào) '}' 作為結(jié)束

注意:格式化字符串字面值中的表達(dá)式:

  • 當(dāng)作正常的 Python 表達(dá)式處理這一點(diǎn)很重要,也很強(qiáng)大
  • 不允許空表達(dá)式
  • lambda 表達(dá)式必須顯式地加上圓括號(hào)
  • 可以包含換行(例如三引號(hào)字符串中)
  • 不能包含注釋
  • 從左到右被求值

如果指定了轉(zhuǎn)換符甩骏,則表達(dá)式求值的結(jié)果會(huì)先轉(zhuǎn)換再格式化窗市。之后使用 Python 內(nèi)置函數(shù) format() 進(jìn)行格式化。

基本用法

name = 'tinyDolphin'
f'my name is {name}'            # 'my name is tinyDolphin'
f'my name is {name!r}'          # "my name is 'tinyDolphin'"

width = 10
precision = 4
value = 12.34567
f'{value:{width}.{precision}}'  # '     12.35'

today = datetime.datetime(year=2019, month=6, day=10)
f'{today:%B %d, %Y}'            # 'June 10, 2019'

number = 1024
f'{number:#0x}'                 # '0x400'

PS:與 str.format() 有一點(diǎn)不同:在 str.format() 中横漏,非數(shù)字索引將自動(dòng)轉(zhuǎn)化為字符串谨设,而f-strings則不會(huì)

kwargs = {'a':1, 'b':2}
# 使用 str.format()
'a={kwargs[a]}'.format(kwargs=kwargs)   # 'a=1'

# 使用 f-string
f"a={kwargs[a]}"                        # × 發(fā)生異常
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'a' is not defined

# 正確使用 f-string
f"a={kwargs['a']}"                      # 'a=1'

總結(jié)

從以下三個(gè)方面進(jìn)行對(duì)比:

速度上:f-string > %-formatting > str.format()
功能上:f-string > str.format() > %-formatting
可讀性: f-string > str.format() > %-formatting

對(duì)于速度驗(yàn)證缎浇,筆者就不在此進(jìn)行驗(yàn)證了扎拣。

推薦使用場(chǎng)景:

%-formatting:Python2 中,由于其性能優(yōu)勢(shì),在涉及輸出的一些操作時(shí)二蓝,優(yōu)先推薦使用
str.format():Python2 中誉券,由于其功能優(yōu)勢(shì),在涉及字符串拼接的一些操作時(shí)刊愚,優(yōu)先推薦使用
f-string:    Python3 中踊跟,強(qiáng)烈推薦使用
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市鸥诽,隨后出現(xiàn)的幾起案子商玫,更是在濱河造成了極大的恐慌,老刑警劉巖牡借,帶你破解...
    沈念sama閱讀 206,126評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件拳昌,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡钠龙,警方通過查閱死者的電腦和手機(jī)炬藤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)碴里,“玉大人沈矿,你說(shuō)我怎么就攤上這事∫б福” “怎么了羹膳?”我有些...
    開封第一講書人閱讀 152,445評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)帝火。 經(jīng)常有香客問我溜徙,道長(zhǎng)湃缎,這世上最難降的妖魔是什么犀填? 我笑而不...
    開封第一講書人閱讀 55,185評(píng)論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮嗓违,結(jié)果婚禮上九巡,老公的妹妹穿的比我還像新娘。我一直安慰自己蹂季,他們只是感情好冕广,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著偿洁,像睡著了一般撒汉。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上涕滋,一...
    開封第一講書人閱讀 48,970評(píng)論 1 284
  • 那天睬辐,我揣著相機(jī)與錄音,去河邊找鬼。 笑死溯饵,一個(gè)胖子當(dāng)著我的面吹牛侵俗,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播丰刊,決...
    沈念sama閱讀 38,276評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼隘谣,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了啄巧?” 一聲冷哼從身側(cè)響起寻歧,我...
    開封第一講書人閱讀 36,927評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎秩仆,沒想到半個(gè)月后熄求,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡逗概,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評(píng)論 2 323
  • 正文 我和宋清朗相戀三年弟晚,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片逾苫。...
    茶點(diǎn)故事閱讀 37,997評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡卿城,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出铅搓,到底是詐尸還是另有隱情瑟押,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評(píng)論 4 322
  • 正文 年R本政府宣布星掰,位于F島的核電站多望,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏氢烘。R本人自食惡果不足惜怀偷,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望播玖。 院中可真熱鬧椎工,春花似錦、人聲如沸蜀踏。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)果覆。三九已至颅痊,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間局待,已是汗流浹背斑响。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評(píng)論 1 260
  • 我被黑心中介騙來(lái)泰國(guó)打工吗讶, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人恋捆。 一個(gè)月前我還...
    沈念sama閱讀 45,423評(píng)論 2 352
  • 正文 我出身青樓照皆,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親沸停。 傳聞我的和親對(duì)象是個(gè)殘疾皇子膜毁,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 一、Python簡(jiǎn)介和環(huán)境搭建以及pip的安裝 4課時(shí)實(shí)驗(yàn)課主要內(nèi)容 【Python簡(jiǎn)介】: Python 是一個(gè)...
    _小老虎_閱讀 5,720評(píng)論 0 10
  • 時(shí)光總是如此易逝 曾經(jīng)的小人呀 如今已經(jīng)變成大人模樣 嬌羞的新娘 是那么的美麗 時(shí)光請(qǐng)善對(duì)這對(duì)新人 時(shí)光不老 他們不散
    雨落無(wú)聲_閱讀 113評(píng)論 0 1
  • 大海撈針---沒處尋 麻雀雖小---五臟俱全 雞蛋破石頭---不自量力 留得青山在---不怕沒柴燒 秀才遇到兵--...
    阿牛g閱讀 317評(píng)論 0 3
  • “能者多勞”在當(dāng)今社會(huì)似乎成了一個(gè)叫他人干活的詞愤钾。比如新入職的小王被同事各種加任務(wù)瘟滨,有人看不過去,同事們就會(huì)來(lái)一...
    醒來(lái)無(wú)味_0172閱讀 56評(píng)論 0 0
  • #白馬聲慢能颁,我自手書# 那年杂瘸,在那個(gè)梅花開滿遍山頭,雪花落滿枝頭的季節(jié)里伙菊,我們相遇了败玉,以一種“同桌”的方式...
    染墨丶若流云閱讀 273評(píng)論 0 0