Python 的”格式化输出”

缘起

打印字符串是编程中经常需要处理的任务, 看路飞学院的视频了解到Python的”格式化输出”功能.

它可以高效地打印多个字符串, 并且进行美化.

定义

“字符串的格式化输出”就是预先制定一个模版, 在这个模版中先空出后续需要输入的内容, 最后, 再将内容填充到这些空出的位置.

那么, 如何让计算机知道哪些位置是需要填写内容的呢?

那么, 就需要约定一些特殊的符号, 标记这些位置. 这些特殊符号被称为”占位符” (很形象, 占住位置. 就像大学图书馆经常看到的, 桌子上放本书, 宣示这个座位已经有人占了)

看一个简单的例子:

  1. print('这款 %s 的售价是 %d 元' % ('iPhone', 4600))

输出结果是

  1. 这款 iPhone 的售价是 4600

其中, %s%d是占位符 (%s表示后续要填入的是 String / 字符串, %d 表示后续要填入的是 digit / 整数), 而后面的 (‘iPhone’, 4600) 则会会在打印时分别填入 %s%d 所处的位置

格式化输出的 3 种方式

1. 最经典

也就是之前那个例子

  1. print('这款 %s 的售价是 %d 元, 我爱 %s.' % ('Mac', 13000, 'Mac'))

输出结果是:

  1. 这款 Mac 的售价是 13000 元, 我爱 Mac.

常用的占位符:

占位符 指代
%s string / 字符串
%d digit / 整数
%f float / 浮点数
%.2f float / 浮点数, 保留 2 位小数
%r raw / 原生的数据 (调试用)

2. 字典

  1. my_dic = {
  2. 'name': 'Mac',
  3. 'price': 13000
  4. }
  5. print('这款 %(name)s 的售价是 %(price)d 元, 我爱 %(name)s.' % my_dic)

输出:

  1. 这款 Mac 的售价是 13000 元, 我爱 Mac.

3. format 函数

  1. print('这款 {0} 的售价是 {1} 元, 我爱 {0}.'.format('Mac', 13000))

输出结果是:

  1. print('这款 {0} 的售价是 {1} 元, 我爱 {0}.'.format('Mac', 13000))

注意:

  • {}里面的序号和 format()里面的内容顺序是对应的, 而且{}里面的序号是从 0 开始的
  • {}里面也可以不写数字, 那么 {} 的数量和format()里面的内容数量要一致:
  1. print('这款 {} 的售价是 {} 元, 我爱 {}.'.format('Mac', 13000, 'Mac'))

输出为:

  1. 这款 Mac 的售价是 13000 元, 我爱 Mac.

python: 格式化字符串f()用法

简介

f-string,亦称为格式化字符串常量(formatted string literals),是Python3.6新引入的一种字符串格式化方法,该方法源于PEP 498 – Literal String Interpolation,主要目的是使格式化字符串的操作更加简便。f-string在形式上是以 fF 修饰符引领的字符串(f'xxx'F'xxx'),以大括号 {} 标明被替换的字段;f-string在本质上并不是字符串常量,而是一个在运行时运算求值的表达式:

While other string literals always have a constant value, formatted strings are really expressions evaluated at run time.
(与具有恒定值的其它字符串常量不同,格式化字符串实际上是运行时运算求值的表达式。)
—— Python Documentation

f-string在功能方面不逊于传统的%-formatting语句str.format()函数,同时性能又优于二者,且使用起来也更加简洁明了,因此对于Python3.6及以后的版本,推荐使用f-string进行字符串格式化。

用法

此部分内容主要参考以下资料:

简单使用

f-string用大括号 {} 表示被替换字段,其中直接填入替换内容:

  1. >>> name = 'Eric'
  2. >>> f'Hello, my name is {name}'
  3. 'Hello, my name is Eric'
  4. >>> number = 7
  5. >>> f'My lucky number is {number}'
  6. 'My lucky number is 7'
  7. >>> price = 19.99
  8. >>> f'The price of this book is {price}'
  9. 'The price of this book is 19.99'
  10. 12345678910111234567891011

表达式求值与函数调用

f-string的大括号 {} 可以填入表达式或调用函数,Python会求出其结果并填入返回的字符串内:

  1. >>> f'A total number of {24 * 8 + 4}'
  2. 'A total number of 196'
  3. >>> f'Complex number {(2 + 2j) / (2 - 3j)}'
  4. 'Complex number (-0.15384615384615388+0.7692307692307692j)'
  5. >>> name = 'ERIC'
  6. >>> f'My name is {name.lower()}'
  7. 'My name is eric'
  8. >>> import math
  9. >>> f'The answer is {math.log(math.pi)}'
  10. 'The answer is 1.1447298858494002'
  11. 1234567891011121312345678910111213

引号、大括号与反斜杠

f-string大括号内所用的引号不能和大括号外的引号定界符冲突,可根据情况灵活切换 '"

  1. >>> f'I am {"Eric"}'
  2. 'I am Eric'
  3. >>> f'I am {'Eric'}'
  4. File "<stdin>", line 1
  5. f'I am {'Eric'}'
  6. ^
  7. SyntaxError: invalid syntax
  8. 12345671234567

'" 不足以满足要求,还可以使用 '''"""

  1. >>> f"He said {"I'm Eric"}"
  2. File "<stdin>", line 1
  3. f"He said {"I'm Eric"}"
  4. ^
  5. SyntaxError: invalid syntax
  6. >>> f'He said {"I'm Eric"}'
  7. File "<stdin>", line 1
  8. f'He said {"I'm Eric"}'
  9. ^
  10. SyntaxError: invalid syntax
  11. >>> f"""He said {"I'm Eric"}"""
  12. "He said I'm Eric"
  13. >>> f'''He said {"I'm Eric"}'''
  14. "He said I'm Eric"
  15. 1234567891011121314151612345678910111213141516

大括号外的引号还可以使用 \ 转义,但大括号内不能使用 \ 转义:

  1. >>> f'''He\'ll say {"I'm Eric"}'''
  2. "He'll say I'm Eric"
  3. >>> f'''He'll say {"I\'m Eric"}'''
  4. File "<stdin>", line 1
  5. SyntaxError: f-string expression part cannot include a backslash
  6. 1234512345

f-string大括号外如果需要显示大括号,则应输入连续两个大括号 {{}}

  1. >>> f'5 {"{stars}"}'
  2. '5 {stars}'
  3. >>> f'{{5}} {"stars"}'
  4. '{5} stars'
  5. 12341234

上面提到,f-string大括号内不能使用 \ 转义,事实上不仅如此,f-string大括号内根本就不允许出现 \。如果确实需要 \,则应首先将包含 \ 的内容用一个变量表示,再在f-string大括号内填入变量名:

  1. >>> f"newline: {ord('\n')}"
  2. File "<stdin>", line 1
  3. SyntaxError: f-string expression part cannot include a backslash
  4. >>> newline = ord('\n')
  5. >>> f'newline: {newline}'
  6. 'newline: 10'
  7. 12345671234567

多行f-string

f-string还可用于多行字符串:

  1. >>> name = 'Eric'
  2. >>> age = 27
  3. >>> f"Hello!" \
  4. ... f"I'm {name}." \
  5. ... f"I'm {age}."
  6. "Hello!I'm Eric.I'm 27."
  7. >>> f"""Hello!
  8. ... I'm {name}.
  9. ... I'm {age}."""
  10. "Hello!\n I'm Eric.\n I'm 27."
  11. 1234567891012345678910

自定义格式:对齐、宽度、符号、补零、精度、进制等

f-string采用 {content:format} 设置字符串格式,其中 content 是替换并填入字符串的内容,可以是变量、表达式或函数等,format 是格式描述符。采用默认格式时不必指定 {:format},如上面例子所示只写 {content} 即可。

关于格式描述符的详细语法及含义可查阅Python官方文档,这里按使用时的先后顺序简要介绍常用格式描述符的含义与作用:

对齐相关格式描述符

格式描述符 含义与作用
< 左对齐(字符串默认对齐方式)
> 右对齐(数值默认对齐方式)
^ 居中

数字符号相关格式描述符

格式描述符 含义与作用
+ 负数前加负号(-),正数前加正号(+
- 负数前加负号(-),正数前不加任何符号(默认)
(空格) 负数前加负号(-),正数前加一个空格

注:仅适用于数值类型。

数字显示方式相关格式描述符

格式描述符 含义与作用
# 切换数字显示方式

注1:仅适用于数值类型。
注2:# 对不同数值类型的作用效果不同,详见下表:

数值类型 不加#(默认) # 区别
二进制整数 '1111011' '0b1111011' 开头是否显示 0b
八进制整数 '173' '0o173' 开头是否显示 0o
十进制整数 '123' '123' 无区别
十六进制整数(小写字母) '7b' '0x7b' 开头是否显示 0x
十六进制整数(大写字母) '7B' '0X7B' 开头是否显示 0X

宽度与精度相关格式描述符

格式描述符 含义与作用
width 整数 width 指定宽度
0width 整数 width 指定宽度,开头的 0 指定高位用 0 补足宽度
width.precision 整数 width 指定宽度,整数 precision 指定显示精度

注1:0width 不可用于复数类型和非数值类型,width.precision 不可用于整数类型。
注2:width.precision 用于不同格式类型的浮点数、复数时的含义也不同:用于 fFeE%precision 指定的是小数点后的位数,用于 gGprecision 指定的是有效数字位数(小数点前位数+小数点后位数)。
注3:width.precision 除浮点数、复数外还可用于字符串,此时 precision 含义是只使用字符串中前 precision 位字符。

示例:

  1. >>> a = 123.456
  2. >>> f'a is {a:8.2f}'
  3. 'a is 123.46'
  4. >>> f'a is {a:08.2f}'
  5. 'a is 00123.46'
  6. >>> f'a is {a:8.2e}'
  7. 'a is 1.23e+02'
  8. >>> f'a is {a:8.2%}'
  9. 'a is 12345.60%'
  10. >>> f'a is {a:8.2g}'
  11. 'a is 1.2e+02'
  12. >>> s = 'hello'
  13. >>> f's is {s:8s}'
  14. 's is hello '
  15. >>> f's is {s:8.3s}'
  16. 's is hel '
  17. 12345678910111213141516171234567891011121314151617

千位分隔符相关格式描述符

格式描述符 含义与作用
, 使用,作为千位分隔符
_ 使用_作为千位分隔符

注1:若不指定 ,_,则f-string不使用任何千位分隔符,此为默认设置。
注2:, 仅适用于浮点数、复数与十进制整数:对于浮点数和复数,, 只分隔小数点前的数位。
注3:_ 适用于浮点数、复数与二、八、十、十六进制整数:对于浮点数和复数,_ 只分隔小数点前的数位;对于二、八、十六进制整数,固定从低位到高位每隔四位插入一个 _(十进制整数是每隔三位插入一个 _)。

示例:

  1. >>> a = 1234567890.098765
  2. >>> f'a is {a:f}'
  3. 'a is 1234567890.098765'
  4. >>> f'a is {a:,f}'
  5. 'a is 1,234,567,890.098765'
  6. >>> f'a is {a:_f}'
  7. 'a is 1_234_567_890.098765'
  8. >>> b = 1234567890
  9. >>> f'b is {b:_b}'
  10. 'b is 100_1001_1001_0110_0000_0010_1101_0010'
  11. >>> f'b is {b:_o}'
  12. 'b is 111_4540_1322'
  13. >>> f'b is {b:_d}'
  14. 'b is 1_234_567_890'
  15. >>> f'b is {b:_x}'
  16. 'b is 4996_02d2'
  17. 12345678910111213141516171234567891011121314151617

格式类型相关格式描述符

基本格式类型

格式描述符 含义与作用 适用变量类型
s 普通字符串格式 字符串
b 二进制整数格式 整数
c 字符格式,按unicode编码将整数转换为对应字符 整数
d 十进制整数格式 整数
o 八进制整数格式 整数
x 十六进制整数格式(小写字母) 整数
X 十六进制整数格式(大写字母) 整数
e 科学计数格式,以 e 表示 ×10^ 浮点数、复数、整数(自动转换为浮点数)
E e 等价,但以 E 表示 ×10^ 浮点数、复数、整数(自动转换为浮点数)
f 定点数格式,默认精度(precision)是6 浮点数、复数、整数(自动转换为浮点数)
F f 等价,但将 naninf 换成 NANINF 浮点数、复数、整数(自动转换为浮点数)
g 通用格式,小数用 f,大数用 e 浮点数、复数、整数(自动转换为浮点数)
G G 等价,但小数用 F,大数用 E 浮点数、复数、整数(自动转换为浮点数)
% 百分比格式,数字自动乘上100后按 f 格式排版,并加 % 后缀 浮点数、整数(自动转换为浮点数)

常用的特殊格式类型:标准库 datetime 给定的用于排版时间信息的格式类型,适用于 datedatetimetime 对象

格式描述符 含义 显示样例
%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. >>> a = 1234
  2. >>> f'a is {a:^#10X}' # 居中,宽度10位,十六进制整数(大写字母),显示0X前缀
  3. 'a is 0X4D2 '
  4. >>> b = 1234.5678
  5. >>> f'b is {b:<+10.2f}' # 左对齐,宽度10位,显示正号(+),定点数格式,2位小数
  6. 'b is +1234.57 '
  7. >>> c = 12345678
  8. >>> f'c is {c:015,d}' # 高位补零,宽度15位,十进制整数,使用,作为千分分割位
  9. 'c is 000,012,345,678'
  10. >>> d = 0.5 + 2.5j
  11. >>> f'd is {d:30.3e}' # 宽度30位,科学计数法,3位小数
  12. 'd is 5.000e-01+2.500e+00j'
  13. >>> import datetime
  14. >>> e = datetime.datetime.today()
  15. >>> f'the time is {e:%Y-%m-%d (%a) %H:%M:%S}' # datetime时间格式
  16. 'the time is 2018-07-14 (Sat) 20:46:02'
  17. 12345678910111213141516171819201234567891011121314151617181920

lambda表达式

f-string大括号内也可填入lambda表达式,但lambda表达式的 : 会被f-string误认为是表达式与格式描述符之间的分隔符,为避免歧义,需要将lambda表达式置于括号 () 内:

  1. >>> f'result is {lambda x: x ** 2 + 1 (2)}'
  2. File "<fstring>", line 1
  3. (lambda x)
  4. ^
  5. SyntaxError: unexpected EOF while parsing
  6. >>> f'result is {(lambda x: x ** 2 + 1) (2)}'
  7. 'result is 5'
  8. >>> f'result is {(lambda x: x ** 2 + 1) (2):<+7.2f}'
  9. 'result is +5.00 '
  10. 1234567891012345678910
  1. <link href="https://csdnimg.cn/release/phoenix/mdeditor/markdown_views-e9f16cbbc2.css" rel="stylesheet">
  2. </div>
  3. 12