Python字符串格式化f-string
Python字符串格式化 f-string
简单介绍
格式字符串字面值 或称 f-string 是标注了 'f'
或 'F'
前缀的字符串字面值。这种字符串可包含替换字段,即以 {}
标注的表达式。其他字符串字面值只是常量,格式字符串字面值则是可在运行时求值的表达式。
基本语法如下:
1 |
|
起源
f-string
是格式化字符串的一种很好的方法。与其他格式化方式相比,它们不仅更易读,更简洁,不易出错,而且速度更快!
3.6 新版功能.
在 3.7 版更改: Python 3.7 以前, 因为实现的问题,不允许在格式字符串字面值表达式中使用
await
表达式与包含async for
子句的推导式。3.8 新版功能: 等号
'='
。'='
用法通常用于调试过程输出变量名称及其存储的值.表达式里含等号 ‘=’ 时,输出内容包括表达式文本、’=’ 、求值结果。输出内容可以保留表达式中左花括号 ‘{‘ 后,及 ‘=’ 后的空格。没有指定格式时,’=’ 默认调用表达式的 repr()。指定了格式时,默认调用表达式的 str(),除非声明了转换字段 ‘!r’。
基本用法
整数格式化
基本格式类型
格式描述符 | 含义与作用 | 适用变量类型 |
---|---|---|
s | 普通字符串格式 | 字符串 |
b | 二进制整数格式 | 整数 |
c | 字符格式,按 unicode 编码将整数转换为对应字符 | 整数 |
d | 十进制整数格式 | 整数 |
o | 八进制整数格式 | 整数 |
x | 十六进制整数格式(小写字母) | 整数 |
X | 十六进制整数格式(大写字母) | 整数 |
e | 科学计数格式,以 e 表示 ×10^ | 浮点数、复数、整数(自动转换为浮点数) |
E | 与 e 等价,但以 E 表示 ×10^ | 浮点数、复数、整数(自动转换为浮点数) |
f | 定点数格式,默认精度(precision)是 6 | 浮点数、复数、整数(自动转换为浮点数) |
F | 与 f 等价,但将 nan 和 inf 换成 NAN 和 INF | 浮点数、复数、整数(自动转换为浮点数) |
g | 通用格式,小数用 f,大数用 e | 浮点数、复数、整数(自动转换为浮点数) |
G | 与 G 等价,但小数用 F,大数用 E | 浮点数、复数、整数(自动转换为浮点数) |
% | 百分比格式,数字自动乘上 100 后按 f 格式排版,并加 % 后缀 | 浮点数、整数(自动转换为浮点数 |
数学符号相关格式描述符
格式描述符 | 含义与作用 |
---|---|
+ |
负数前加负号(- ),正数前加正号(+ ) |
- |
负数前加负号(- ),正数前不加任何符号(默认) |
(空格) |
负数前加负号(- ),正数前加一个空格 |
# |
切换数字显示方式, 用于控制是否显示进制前缀, 例如二进制’0b’, 八进制的’0o’和十六进制的’0x’ |
注:
空格或其他符合也可作为宽度占位符(例如下文对齐和填充字符
部分用法).
千分位占位符也可以使用_
.
8 16 进制的占位符如果使用大写格式, 那么打印出的进制前缀也会变为相应大写格式.
1 |
|
前导0、统一宽度、千分位、八进制、十六进制、带前缀大写十六进制、二进制
No.0000: 64972: 64,972: 176714: fdcc: 0XFDCC:00001111110111001100
No.0001: 92916: 92,916: 265364: 16af4: 0X16AF4:00010110101011110100
No.0002: 99560: 99,560: 302350: 184e8: 0X184E8:00011000010011101000
No.0003: 23767: 23,767: 56327: 5cd7: 0X5CD7:00000101110011010111
No.0004: 33628: 33,628: 101534: 835c: 0X835C:00001000001101011100
No.0005: 710: 710: 1306: 2c6: 0X2C6:00000000001011000110
No.0006: 47749: 47,749: 135205: ba85: 0XBA85:00001011101010000101
No.0007: 56646: 56,646: 156506: dd46: 0XDD46:00001101110101000110
No.0008: 22526: 22,526: 53776: 57fe: 0X57FE:00000101011111111110
No.0009: 34263: 34,263: 102727: 85d7: 0X85D7:00001000010111010111
浮点数格式化
1 |
|
前导0、统一宽度右对齐、千分位、小数点后固定位数、百分比
000004842.42: 4842.417: 4,842.42: 484241.7%
000003106.63: 3106.634: 3,106.63: 310663.4%
000009213.21: 9213.205: 9,213.21: 921320.5%
000008040.67: 8040.671: 8,040.67: 804067.1%
000006086.32: 6086.322: 6,086.32: 608632.2%
000005251.10: 5251.105: 5,251.10: 525110.5%
000008219.07: 8219.072: 8,219.07: 821907.2%
000007046.32: 7046.325: 7,046.32: 704632.5%
000000192.89: 192.888: 192.89: 19288.8%
000006164.21: 6164.208: 6,164.21: 616420.8%
时间格式化
常用的特殊格式类型:标准库 datetime
给定的用于排版时间信息的格式类型,适用于 date
、datetime
和 time
对象
格式描述符 | 含义 | 显示样例 |
---|---|---|
%a | 星期几(缩写) | ‘Sun’ |
%A | 星期几(全名) | ‘Sunday’ |
%w | 星期几(数字,0 是周日,6 是周六) | ‘0’ |
%u | 星期几(数字,1 是周一,7 是周日) | ‘7’ |
%d | 日(数字,以 0 补足两位) | ‘07’ |
%b | 月(缩写) | ‘Aug’ |
%B | 月(全名) | ‘August’ |
%m | 月(数字,以 0 补足两位) | ‘08’ |
%y | 年(后两位数字,以 0 补足两位) | ‘14’ |
%Y | 年(完整数字,不补零) | ‘2014’ |
%H | 小时(24 小时制,以 0 补足两位) | ‘23’ |
%I | 小时(12 小时制,以 0 补足两位) | ‘11’ |
%p | 上午/下午 | ‘PM’ |
%M | 分钟(以 0 补足两位) | ‘23’ |
%S | 秒钟(以 0 补足两位) | ‘56’ |
%f | 微秒(以 0 补足六位) | ‘553777’ |
%z | UTC 偏移量(格式是 ±HHMM[SS],未指定时区则返回空字符串) | ‘+1030’ |
%Z | 时区名(未指定时区则返回空字符串) | ‘EST’ |
%j | 一年中的第几天(以 0 补足三位) | ‘195’ |
%U | 一年中的第几周(以全年首个周日后的星期为第 0 周,以 0 补足两位) | ‘27’ |
%w | 一年中的第几周(以全年首个周一后的星期为第 0 周,以 0 补足两位) | ‘28’ |
%V | 一年中的第几周(以全年首个包含 1 月 4 日的星期为第 1 周,以 0 补足两位) | ‘28’ |
1 |
|
'the time is 2022-12-29 (Thu) 22:23:58'
格式化打印各种数据类型
1 |
|
str: 'animal' in a line
int: 3387
float: -31220036492.2902
bool: False
list: ['expect', 'shake', 'there']
dict: {'yard': 7553}
set: {106}
tuple(2603, 8262, 3320)
对齐和填充字符
1 |
|
----------左中右对齐----------
|James | James | James|
|Luis | Luis | Luis|
|Keith | Keith | Keith|
|Kimberly | Kimberly | Kimberly|
|Lauren | Lauren | Lauren|
|Gregory | Gregory | Gregory|
|Daniel | Daniel | Daniel|
|Adam | Adam | Adam|
|Emily | Emily | Emily|
|Jessica | Jessica | Jessica|
----------填充字符----------
0...........James
1............Luis
2...........Keith
3........Kimberly
4..........Lauren
5.........Gregory
6..........Daniel
7............Adam
8...........Emily
9.........Jessica
此处f"\n{'-' * 10}{'填充字符'}{'-' * 10}"
可以用f"\n{'填充字符':-^24}"
替代, f-string的优雅之处在此刻尽显
1 |
|
-------------填充字符并居中对齐-------------
[-----------------James------------------]
[------------------Luis------------------]
[-----------------Keith------------------]
[----------------Kimberly----------------]
[-----------------Lauren-----------------]
[----------------Gregory-----------------]
[-----------------Daniel-----------------]
[------------------Adam------------------]
[-----------------Emily------------------]
[----------------Jessica-----------------]
调试阶段输出变量名及其内容
1 |
|
a=12, b=34, a*a+b*b=1300
嵌套调用
1 |
|
3
3.1
3.14
3.142
3.1416
3.14159
3.141593
3.1415927
3.14159265
3.141592654
多行f-string
1 |
|
"Hi I'm leo. My age is 21. python is the best language."
任意表达式
f-string
是在运行时进行渲染的, 因此可以使用表达式, 函数…
这将使您的代码变得更加优雅
细心的您可能会发现, 上文有部分使用了此特性
1 |
|
My name is leo. Tomorrow my age will be 22
您甚至可以使用f-string
来格式化您打印的对象
就像这样:
1 |
|
<Person> leo, 21, student
注意
在使用 f-string 时大括号内所用的引号不能和大括号外的引号定界符冲突, 根据情况切换’和”, 如果仍不满足需求可以使用’’’和”””
1 |
|
1 |
|
当您想要在f-string中打印{}
时, 需要使用{{}}
, 此时双括号内部的内容将不会被渲染
1 |
|
conversion 用法
指定了转换符时,表达式求值的结果会先转换,再格式化。转换符 '!s'
调用 str()
转换求值结果,'!r'
调用 repr()
,'!a'
调用 ascii()
。
默认情况下,f-string 将使用 str(),但如果包含转换标志,则可以确保它们使用 repr()
1 |
|
Wang Leo is 21.
Wang Leo is 21. made by repr!
ascii(object)
与 repr() 类似,返回一个字符串,表示对象的可打印形式,但在 repr() 返回的字符串中,非 ASCII 字符会用 \x、\u 和 \U 进行转义。生成的字符串类似于 Python 2 中 repr() 的返回结果。
ascii(object)
与 repr() 类似,返回一个字符串,表示对象的可打印形式,但在 repr() 返回的字符串中,非 ASCII 字符会用 \x、\u 和 \U 进行转义。生成的字符串类似于 Python 2 中 repr() 的返回结果。
!a 用法
1 |
|
'\u963f\u591a \U0001f525 hold the door'
'\U0001f4af \u2714 \U0001f308'
1 |
|
Wang Leo is 21.
Wang Leo is 21. made by repr!
1 |
|
'\u963f\u591a \U0001f525 hold the door'
'\U0001f4af \u2714 \U0001f308'
旧时代的格式化字符串
在 Python 3.6 之前,有两种将 Python 表达式嵌入到字符串文本中进行格式化的主要方法:
%-formatting
str.format()
这两种方法都具有一定的局限性
从 Python 3.6 开始,f-string
作为格式化字符串的一种很好的新方法。与其他格式化方式相比,
不仅更易读,更简洁,不易出错,而且速度更快!
%-formatting
字符串支持使用%
进行格式化, 例如:
1 |
|
Hello, I'm leo.
为什么 %-formatting
不好用
上面刚刚看到的代码示例足够易读。
但是,一旦你开始使用几个参数和更长的字符串,你的代码将很快变得不太容易阅读。
这种格式不是很好,因为它是冗长的,会导致错误,比如不能正确显示元组或字典, 不方便找出每个占位符对应的变量。
幸运的是,未来有更光明的日子 — 3.6 版本, f-string
出现了。
如你所见, 使用f-string
后代码变得更加直观
1 |
|
Hello, Eric Idle. You are 08. You are a comedian. You were a member of Monty Python.
Hello, Eric Idle. You are 08. You are a comedian. You were a member of Monty Python.
str.format()
这种字符串格式化方式早在 2.6 版本就已经引入
str.format()
是对%-formatting
的改进。它使用正常的函数调用语法,并且可以通过对要转换为字符串的对象的**format **()方法进行扩展。
使用 str.format(),被替换字段用大括号标记:
1 |
|
Hello, leo. I'm 08.
并且您可以使用索引来决定使用变量的顺序
1 |
|
Hello, leo. my info is: leo-08.
与 %-formatting
相比, str.format()
完全可以说是一个升级版本.
但当处理多个参数和更长的字符串时, 仍存在过于冗长, 易读性差等问题
1 |
|
Hello, Eric Idle. You are 74. You are a comedian. You were a member of Monty Python.
性能
f-string 中的 f 也可以代表“速度快”。
f-string 比%-formatting 和 str.format()都快。正如你已经看到的,f-string 是运行时渲染的表达式,而不是常量值。以下是文档摘录:
“F-strings provide a way to embed expressions inside string literals, using a minimal syntax. It should be noted that an f-string is really an expression evaluated at run time, not a constant value. In Python source code, an f-string is a literal string, prefixed with f, which contains expressions inside braces. The expressions are replaced with their values.” (Source)
在运行时,大括号内的表达式将在其自己的作用域中进行求值,然后将其与其余字符串组合在一起。
以下是性能测试:
1 |
|
227 ns ± 4.81 ns per loop (mean ± std. dev. of 7 runs, 1,000,000 loops each)
1 |
|
278 ns ± 3.37 ns per loop (mean ± std. dev. of 7 runs, 1,000,000 loops each)
1 |
|
17.7 ns ± 0.0882 ns per loop (mean ± std. dev. of 7 runs, 100,000,000 loops each)
如您所见, f-string, %-formatting 和 str.format()中 f-string 是最快的
参考链接
(排名不分先后)
Python 格式化字符串 f-string f”{}{}{}”详细介绍
made by leo wang
date: 2022/12/30