Python中str.format()方法的具體使用
1. 術語說明
- str.format() 方法通過字符串中的花括號 {} 來識別替換字段 replacement field,從而完成字符串的格式化。
- 替換字段 由字段名 field name 和轉換字段 conversion field 以及格式說明符 format specifier 組成,即一般形式為 {字段名!轉換字段:格式說明符}。
- 字段名分為簡單字段名 simple field name 和復合字段名 compound field name。
- 而轉換字段和格式說明符都是可選的。
2. 簡單字段名
2.1 簡單字段名的說明
簡單字段名有三種寫法:
- 省略不寫 {}
- 數字 {十進制非負整數}
- 變量名 {合法的Python標識符}
2.2 省略字段名
花括號內省略字段名,傳遞位置參數。
- 替換字段形式: {}
- 註意:花括號個數可以少於位置參數的個數,反之不然。
# 省略字段名傳遞位置參數 print('我叫{},今年{}歲。'.format('小明', 18)) """ 我叫小明,今年18歲。 """ # 花括號個數可以少於位置參數的個數 print('我愛吃{}和{}。'.format('香蕉', '蘋果', '大鴨梨')) """ 我愛吃香蕉和蘋果。 """ # 花括號個數多於位置參數的個數則會報錯 # print('我還吃{}和{}。'.format('西紅柿')) """ IndexError: tuple index out of range """
2.3 數字形式的簡單字段名
可以通過數字形式的簡單字段名傳遞位置參數。
- 數字必須是大於等於 0 的整數。
- 帶數字的替換字段可以重復使用。
- 數字形式的簡單字段名相當於把 format 中的所有位置參數整體當作一個元組,通過字段名中的數字進行取值。
- 即 {0} 等價於 tuple[0],所以花括號內的數字不能越界。
# 通過數字形式的簡單字段名傳遞位置參數 print('身高{0},傢住{1}。'.format(1.8, '銅鑼灣')) """ 身高1.8,傢住銅鑼灣 """ # 數字形式的簡單字段名可以重復使用。 print('我愛{0}。\n她今年{1}。\n{0}也愛我。'.format('阿香', 17)) """ 我愛阿香。 她今年17。 阿香也愛我。 """ # 體會把所有位置參數整體當成元組來取值 print('阿香愛吃{1}、{3}和{0}。'.format( '榴蓮', '臭豆腐', '皮蛋', '鯡魚罐頭', '螺獅粉')) """ 阿香愛吃臭豆腐、鯡魚罐頭和榴蓮。 """ # 嘗試一下越界錯誤 # print('{1}'.format('錯誤用法')) """ IndexError: tuple index out of range """
2.4 變量名形式的簡單字段名
使用變量名形式的簡單字段名傳遞關鍵字參數。
關鍵字參數的位置可以隨意調換。
# 使用變量名形式的簡單字段名傳遞關鍵字參數 print('我大哥是{name},今年{age}歲。'.format(name='阿飛', age=20)) """ 我大哥是阿飛,今年20歲。 """ # 關鍵字參數的順序可以隨意調換 print('我大哥是{name},今年{age}歲。'.format(age=20, name='阿飛')) """ 我大哥是阿飛,今年20歲。 """
2.5 簡單字段名的混合使用
- 混合使用數字形式和變量名形式的字段名,可以同時傳遞位置參數和關鍵字參數。
- 關鍵字參數必須位於位置參數之後。
- 混合使用時可以省略數字。
- 省略字段名 {} 不能和數字形式的字段名 {非負整數} 同時使用。
# 混合使用數字形式和變量名形式的字段名 # 可以同時傳遞位置參數和關鍵字參數 print('這是一個關於{0}、{1}和{girl}的故事。'.format( '小明', '阿飛', girl='阿香')) """ 這是一個關於小明、阿飛和阿香的故事。 """ # 但是關鍵字參數必須位於位置參數之後 # print('這是一個關於{0}、{1}和{girl}的故事。'.format( # '小明', girl='阿香' , '阿飛')) """ SyntaxError: positional argument follows keyword argument """ # 數字也可以省略 print('這是一個關於{}、{}和{girl}的故事。'.format( '小明', '阿飛', girl='阿香')) # 但是省略字段名不能和數字形式的字段名同時出現 # print('這是一個關於{}、{1}和{girl}的故事。'.format( # '小明', '阿飛', girl='阿香')) """ ValueError: cannot switch from automatic field numbering to manual field specification """
2.6 使用元組和字典傳參
str.format() 方法還可以使用 *元組 和 **字典 的形式傳參,兩者可以混合使用。
位置參數、關鍵字參數、*元組 和 **字典 也可以同時使用,但是要註意,位置參數要在關鍵字參數前面,*元組 要在 **字典 前面。
# 使用元組傳參 infos = '鋼鐵俠', 66, '小辣椒' print('我是{},身價{}億。'.format(*infos)) """ 我是鋼鐵俠,身傢66億。 """ print('我是{2},身價{1}億。'.format(*infos)) """ 我是小辣椒,身傢66億。 """ # 使用字典傳參 venom = {'name': '毒液', 'weakness': '火'} print('我是{name},我怕{weakness}。'.format(**venom)) """ 我是毒液,我怕火。 """ # 同時使用元組和字典傳參 hulk = '綠巨人', '拳頭' captain = {'name': '美國隊長', 'weapon': '盾'} print('我是{}, 我怕{weapon}。'.format(*hulk, **captain)) print('我是{name}, 我怕{1}。'.format(*hulk, **captain)) """ 我是綠巨人, 我怕盾。 我是美國隊長, 我怕拳頭。 """ # 同時使用位置參數、元組、關鍵字參數、字典傳參 # 註意: # 位置參數要在關鍵字參數前面 # *元組要在**字典前面 tup = '鷹眼', dic = {'weapon': '箭'} text = '我是{1},我怕{weakness}。我是{0},我用{weapon}。' text = text.format( *tup, '黑寡婦', weakness='男人', **dic) print(text) """ 我是黑寡婦,我怕男人。我是鷹眼,我用箭。 """
3. 復合字段名
3.1 復合字段名的說明
同時使用瞭數字和變量名兩種形式的字段名就是復合字段名。
復合字段名支持兩種操作符:
- [] 方括號
- . 點號
3.2 . 點號的使用
傳遞位置參數
- 替換字段形式:{數字.屬性名}
- 隻有一個替換字段的時候可以省略數字
# 復合字段名中使用點號傳遞對象屬性 class Person: def __init__(self, name, addr): self.name = name self.addr = addr p = Person('辣妹子', '重慶') # 點號用法:傳遞位置參數 print('我是{0.name},傢在{0.addr}。'.format(p)) """ 我是辣妹子,傢在重慶。 """ # 當隻有一個替換字段的時候可以省略數字 print('{.name}辣!'.format(p)) """ 辣妹子辣! """ # 試試傳遞文件對象的屬性 f = open('out.txt', 'w') print('文件名為:{.name}'.format(f)) f.close() """ 文件名為:out.txt """
傳遞關鍵字參數
替換字段形式:{關鍵字參數名.屬性名}
# 點號用法:傳遞關鍵字參數 print('我是{girl.name},傢在{girl.addr}。'.format(girl=p)) # 和上一句等價 print('我是{p.name},傢在{p.addr}。'.format(p=p)) """ 我是辣妹子,傢在重慶。 我是辣妹子,傢在重慶。 """
3.3 [] 方括號的使用
傳遞位置參數
- 用列表傳遞位置參數
- 用元組傳遞位置參數
- 用字典傳遞位置參數
# 方括號用法:用列表傳遞位置參數 infos = ['阿星', 9527] food = ['霸王花', '爆米花'] print('我叫{0[0]},警號{0[1]},愛吃{1[0]}。'.format( infos, food)) """ 我叫阿星,警號9527,愛吃霸王花。 """ # 方括號用法:用元組傳遞位置參數 food = ('僵屍', '腦子') print('我叫{0[0]},年齡{1},愛吃{0[1]}。'.format( food, 66)) """ 我叫僵屍,年齡66,愛吃腦子。 """ # 方括號用法:用字典傳遞位置參數 dic = dict(name='阿星', pid=9527) print('我是{[name]}!'.format( dic)) # 多個替換字段,不能省略數字 print('我是{0[name]},警號{0[pid]}。'.format( dic)) """ 我是阿星! 我是阿星,警號9527。 """
傳遞關鍵字參數
- 用列表傳遞關鍵字參數
- 用元組傳遞關鍵字參數
- 用字典傳遞關鍵字參數
# 方括號用法:傳遞關鍵字參數 names = ['皮卡丘'] dic = {'name': '妙蛙花'} skills = ('十萬伏特', '飛葉快刀') text = '我是{names[0]},我會{skills[0]}。我是{dic[name]},我會{skills[1]}。' text = text.format(names=names, skills=skills, dic=dic) print(text) """ 我是皮卡丘,我會十萬伏特。我是妙蛙花,我會飛葉快刀。 """
4. 轉換字段
轉換字段 conversion field 的取值有三種,前面要加 !:
- s:傳遞參數之前先對參數調用 str()
- r:傳遞參數之前先對參數調用 repr()
- a:傳遞參數之前先對參數調用 ascii()
ascii() 函數類似 repr() 函數,返回一個可以表示對象的字符串。
但是對於非 ASCII 字符,使用 \x,\u 或者 \U 轉義。
# 轉換字段 print('I am {!s}!'.format('Bruce Lee 李小龍')) print('I am {!r}!'.format('Bruce Lee 李小龍')) print('I am {!a}!'.format('Bruce Lee 李小龍')) """ I am Bruce Lee 李小龍! I am 'Bruce Lee 李小龍'! I am 'Bruce Lee \u674e\u5c0f\u9f99'! """
5. 格式說明符
在替換字段中,格式說明符前面有一個冒號 : :
{字段名!轉換字段:格式說明符}
其中格式說明符本身可以是一個字段名,比如:
print('{0:{1}}'.format(3.14159, '.4f')) """ 3.1416 """
5.1 標準格式說明符的格式
如果不通過重寫 __format__ 方法來進行自定義的話,標準格式說明符的形式如下。其中方括號是可選的。
[[fill]align][sign][#][0][width][grouping_option][.precision][type]
中文形式可以寫作:
[[填充]對齊方式][正負號][#][0][寬度][分組選項][.精度][類型碼]
5.2 填充與對齊方式
填充:
- 隻能是一個字符
- 不指定默認用空格填充
- 如果指定填充字符,則必須要同時指定對齊方式
對齊方式的取值:
- <:左對齊
- >:右對齊
- ^:居中
- =:在正負號(如果有的話)和數字之間填充,該對齊選項僅對數字類型有效。它可以輸出類似 +0000120 這樣的字符串。
註意:
如果不給定最小寬度 width,對齊方式毫無意義。
5.3 正負號
- 正負號選項僅對數字類型生效
- 取值有三種:
- + 正數前面添加正號,負數前面添加負號
- – 僅在負數前面添加負號(默認行為)
- 空格:正數前面需要添加一個空格,以便與負數對齊
# 正負號 print('{:哈=+8.2f}'.format(3.14159)) print('{:哈=+8.2f}'.format(-3.14159)) print('{:哈=+8.2f}'.format(0)) print('{:哈=+8.2f}'.format(-0)) """ +哈哈哈3.14 -哈哈哈3.14 +哈哈哈0.00 +哈哈哈0.00 """
5.4 # 號
# 號:
- 給u二進制數加上 0b 前綴
- 給八進制數加上 0o 前綴
- 給十六進制數加上 0x 前綴
5.5 最小寬度
最小寬度 width:
- 如果不指定,最小字段寬度由內容決定,與內容相等
- 如果最小寬度前面有一個前導 0,意味著用 0 填充
- 這等價於指定瞭 0= 的填充和對齊方式
5.6 分組選項
分組選項 grouping_option 的取值有兩種:
逗號 ,:使用逗號對數字以千為單位進行分隔。n 類型的數字可以使用本地化的分隔符。
n類型在本機無法使用分組選項 ,
原因可能是中文沒有數字分隔符
# n 類型使用本地化的分組選項 , # 此項報錯,我懷疑是因為中文沒有數字的分隔符 # print('數字:{0:,n}'.format(6666)) """ ValueError: Cannot specify ',' with 'n'. """ # 使用 d 類型確實是可以的 print('數字:{0:,d}'.format(6666)) """ 數字:6,666 """
下劃線 _:使用下劃線對浮點數和 d 類型的整數以千為單位進行分隔。對於 b、o、x 和 X 類型,每四位插入一個下劃線,其他類型都會報錯。
# 分組選項 _ 作用於 b 類型 print('數字:{0:_b}'.format(0b100111011)) """ 數字:1_0011_1011 """ # 分組選項 _ 作用於 o 類型 print('數字:{0:_o}'.format(0o426754316)) """ 數字:4_2675_4316 """ # 分組選項 _ 作用於 x 類型 print('數字:{0:_x}'.format(0x2a846e98d)) """ 數字:2_a846_e98d """ # 分組選項 _ 作用於 X 類型 print('數字:{0:_X}'.format(0X2a846e98d)) """ 數字:2_A846_E98D """ # 分組選項 _ 作用於其他類型(比如 n 類型) # print('字符串:{0:_n}'.format(1234567)) """ ValueError: Cannot specify ',' with 'n'. """
5.7 精度
精度:
- 精度指定瞭小數點後面要展示多少位小數
- 對於非數字類型,精度指定瞭最大字段寬度
- 整數類型不能指定精度
# 對於非數字類型,精度指定最大字段寬度 print('{0:.3}'.format('哇哈哈哈哈哈')) """ 哇哈哈 """ # 整數類型不能指定精度 print('{:.3d}'.format(666)) """ ValueError: Precision not allowed in integer format specifier """
5.8 類型碼
類型碼可以分為三大類:
- 字符串類型
- 整數類型
- 浮點數類型
5.8.1 字符串類型
s 字符串類型。這是字符串的默認類型,可以省略。
None 不指定類型。同 s 類型。
# s 類型 print('{0:s}'.format('略略略')) # s 類型可以省略 print('{0:}'.format('略略略')) """ 略略略 略略略 """
5.8.2 整數類型
b 二進制。
# b 類型:二進制 print('{0:b}'.format(3)) """ 11 """
c 字符。把整數轉換為相應的 Unicode 字符,然後再打印。
# c 類型:把整數轉換成 unicode 字符 print('{:c}'.format(97)) """ a """
d 十進制整數。
# d 類型:十進制整數 print('{:d}'.format(666)) """ 666 """
o 八進制數。
# o 類型:八進制數 print('{:o}'.format(10)) """ 12 """
x 十六進制數,a 到 f 小寫。
# x 類型:十六進制數,a到f小寫 print('{:x}'.format(15)) """ f """
X 十六進制數,A 到 F 大寫。
# X 類型:十六進制數,A到F大寫 print('{:X}'.format(15)) """ F """
n 數字 number 類型,與 d 相同,隻不過它會使用本地化的數字分隔符。
經試驗,在本機為 n 類型指定任何分組選項(, 和 _)都會報錯。
ValueError: Cannot specify ',' with 'n'.
# n 類型:與d相同,會插入本地化的分隔符 print('{:n}'.format(66666)) # 經試驗,本機無法為 n 指定任何分組選項(,_) # print('{:,n}'.format(66666)) """ ValueError: Cannot specify ',' with 'n'. """ # print('{:_n}'.format(66666)) """ ValueError: Cannot specify ',' with 'n'. """
None 不指定類型,與 d 相同。
5.8.3 浮點數類型
e 科學記數法,用 e 來表示指數。默認精度為 6 位
# e 類型:科學記數法 # 默認精度為 6 位 print('{:e}'.format(1234567.1234567)) """ 1.234567e+06 """
E 與 e 相同,但是使用大寫的 E 表示指數。
# E 類型:與 e 相同,用大寫 E 表示指數 # 默認精度為 6 位 print('{:E}'.format(1234567.1234567)) # 修改精度為 10 位 print('{:.10E}'.format(1234567.1234567)) """ 1.234567E+06 1.2345671235E+06 """
f 定點記法,默認精度為 6。
# f 類型 # 默認精度為 6 位 print('{:f}'.format(1234567.1234567)) """ 1234567.123457 """
F 定點記法,同 f,但是會把 nan 轉換成 NAN,把 inf 轉換成 INF
# F 類型 nan = float('nan') inf = float('inf') print('{:F}\n{:F}'.format(nan, inf)) """ NAN INF """
g 通用 general 格式。自動轉換到 e 或者 f 格式,具體的轉換規則在此省略。正無窮、負無窮、正零、負零和非數字分別顯示為 inf,-inf,0,-0,nan。指定精度為 0 時等價於精度為 1。默認精度為 6 位。
# g 類型 print('{:g}'.format(1234567.1234567)) print('{:g}'.format(1234.1234)) """ 1.23457e+06 1234.12 """
G 通用 general 格式。自動轉換到 E 或者 F 格式,轉換規則同上,相應表示方式換成大寫。
# g 類型 print('{:g}'.format(1234567.1234567)) print('{:g}'.format(1234.1234)) """ 1.23457e+06 1234.12 """
n 數字 number 類型。跟 g 一樣,隻不過用本地化的分隔符來分隔數字。
# n 類型 print('{:n}'.format(1234567.1234567)) print('{:n}'.format(1234.1234)) """ 1.23457E+06 1234.12 """ # 經試驗,本機指定分組選項會報錯 # print('{:,n}'.format(1234.1234)) """ ValueError: Cannot specify ',' with 'n'. """ # print('{:_n}'.format(1234.1234)) """ ValueError: Cannot specify ',' with 'n'. """
% 百分號類型。會將數字乘以 100,然後以 f 定點 fixed-point 格式顯示,最後加上一個百分號 %。
# % 類型 print('{:%}'.format(1)) """ 100.000000% """
None 不指定類型。輸出效果類似調用 str() 函數。
6. 補充說明
輸出花括號需要用花括號本身來轉義
# 打印花括號需要使用花括號轉義 print('{{{}}}'.format('張無忌')) """ {張無忌} """
對象可以自定義格式說明符來替換標準格式說明符,比如 datetime 類。
from datetime import datetime print("Today is: {0:%a %b %d %H:%M:%S %Y}".format(datetime.now())) """ 今天是:Thu Nov 15 13:05:09 2018 """
參考資料
Format String Syntax
PEP 3101 – Advanced String Formatting
Python format 格式化函數
Python之format詳解
Python高級編程
到此這篇關於Python中str.format()方法的具體使用的文章就介紹到這瞭,更多相關Python str.format()內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!