• 沒有找到結果。

设置字符串的格式:完整版

使用字符串

3.3 设置字符串的格式:完整版

4 5

15 6 7 8 9 10 11 12 13 14

>>> "Euler's constant is roughly {e}.".format(e=e)

"Euler's constant is roughly 2.718281828459045."

3.3 设置字符串的格式:完整版

字符串格式设置涉及的内容很多,因此即便是这里的完整版也无法全面探索所有的细节,而 只是介绍主要的组成部分。这里的基本思想是对字符串调用方法format,并提供要设置其格式的 值 。 字 符 串 包 含 有 关 如 何 设 置 格 式 的 信 息 , 而 这 些 信 息 是 使 用 一 种 微 型 格 式 指 定 语 言

(mini-language)指定的。每个值都被插入字符串中,以替换用花括号括起的替换字段。要在最 终结果中包含花括号,可在格式字符串中使用两个花括号(即{{或}})来指定。

>>> "{{ceci n'est pas une replacement field}}".format()

"{ceci n'est pas une replacement field}"

在格式字符串中,最激动人心的部分为替换字段。替换字段由如下部分组成,其中每个部分 都是可选的。

字段名:索引或标识符,指出要设置哪个值的格式并使用结果来替换该字段。除指定值 外,还可指定值的特定部分,如列表的元素。

转换标志:跟在叹号后面的单个字符。当前支持的字符包括r(表示repr)、s(表示str)

和a(表示ascii)。如果你指定了转换标志,将不使用对象本身的格式设置机制,而是使

用指定的函数将对象转换为字符串,再做进一步的格式设置。

格式说明符:跟在冒号后面的表达式(这种表达式是使用微型格式指定语言表示的)。格 式说明符让我们能够详细地指定最终的格式,包括格式类型(如字符串、浮点数或十六 进制数),字段宽度和数的精度,如何显示符号和千位分隔符,以及各种对齐和填充方式。

下面详细介绍其中的一些要素。

3.3.1 替换字段名

在最简单的情况下,只需向format提供要设置其格式的未命名参数,并在格式字符串中使用 未命名字段。此时,将按顺序将字段和参数配对。你还可给参数指定名称,这种参数将被用于相 应的替换字段中。你可混合使用这两种方法。

>>> "{foo} {} {bar} {}".format(1, 2, bar=4, foo=3) '3 1 4 2'

还可通过索引来指定要在哪个字段中使用相应的未命名参数,这样可不按顺序使用未命名 参数。

>>> "{foo} {1} {bar} {0}".format(1, 2, bar=4, foo=3) '3 2 4 1'

然而,不能同时使用手工编号和自动编号,因为这样很快会变得混乱不堪。

你并非只能使用提供的值本身,而是可访问其组成部分(就像在常规Python代码中一样),

如下所示:

>>> fullname = ["Alfred", "Smoketoomuch"]

>>> "Mr {name[1]}".format(name=fullname) 'Mr Smoketoomuch'

>>> import math

>>> tmpl = "The {mod.__name__} module defines the value {mod.pi} for π"

>>> tmpl.format(mod=math)

'The math module defines the value 3.141592653589793 for π'

如你所见,可使用索引,还可使用句点表示法来访问导入的模块中的方法、属性、变量和函 数(看起来很怪异的变量__name__包含指定模块的名称)。

3.3.2 基本转换

指定要在字段中包含的值后,就可添加有关如何设置其格式的指令了。首先,可以提供一个 转换标志。

>>> print("{pi!s} {pi!r} {pi!a}".format(pi="π")) π 'π' '\u03c0'

上述三个标志(s、r和a)指定分别使用str、repr和ascii进行转换。函数str通常创建外观 普通的字符串版本(这里没有对输入字符串做任何处理)。函数repr尝试创建给定值的Python表 示(这里是一个字符串字面量)。函数ascii创建只包含ASCII字符的表示,类似于Python 2中的 repr。

你还可指定要转换的值是哪种类型,更准确地说,是要将其视为哪种类型。例如,你可能提 供一个整数,但将其作为小数进行处理。为此可在格式说明(即冒号后面)使用字符f(表示定 点数)。

>>> "The number is {num}".format(num=42) 'The number is 42'

>>> "The number is {num:f}".format(num=42) 'The number is 42.000000'

你也可以将其作为二进制数进行处理。

>>> "The number is {num:b}".format(num=42) 'The number is 101010'

这样的类型说明符有多个,完整的清单见表3-1。

表3-1 字符串格式设置中的类型说明符

类型 含 义

b 将整数表示为二进制数

c 将整数解读为Unicode码点

d 将整数视为十进制数进行处理,这是整数默认使用的说明符

e 使用科学表示法来表示小数(用e来表示指数)

E 与e相同,但使用E来表示指数

1

>>> "{num:10}".format(num=3) ' 3'

>>> "{name:10}".format(name="Bob") 'Bob '

如你所见,数和字符串的对齐方式不同。对齐将在下一节介绍。

精度也是使用整数指定的,但需要在它前面加上一个表示小数点的句点。

>>> "Pi day is {pi:.2f}".format(pi=pi) 'Pi day is 3.14'

这里显式地指定了类型f,因为默认的精度处理方式稍有不同(相关的规则请参阅“Python 库参考手册”)。当然,可同时指定宽度和精度。

>>> "{pi:10.2f}".format(pi=pi) ' 3.14'

实际上,对于其他类型也可指定精度,但是这样做的情形不太常见。

>>> "{:.5}".format("Guido van Rossum") 'Guido'

最后,可使用逗号来指出你要添加千位分隔符。

>>> 'One googol is {:,}'.format(10**100)

'One googol is 10,000,000,000,000,000,000,000,000,000,000,000,000,000,000,000,000,000,000,00 0,000,000,000,000,000,000,000,000,000,000,000,000,000,000'

同时指定其他格式设置元素时,这个逗号应放在宽度和表示精度的句点之间

3.3.4 符号、对齐和用 0 填充

有很多用于设置数字格式的机制,比如便于打印整齐的表格。在大多数情况下,只需指定宽 度和精度,但包含负数后,原本漂亮的输出可能不再漂亮。另外,正如你已看到的,字符串和数 的默认对齐方式不同。在一栏中同时包含字符串和数时,你可能想修改默认对齐方式。在指定宽 度和精度的数前面,可添加一个标志。这个标志可以是零、加号、减号或空格,其中零表示使用 0来填充数字。

>>> '{:010.2f}'.format(pi) '0000003.14'

要指定左对齐、右对齐和居中,可分别使用<、>和^。

>>> print('{0:<10.2f}\n{0:^10.2f}\n{0:>10.2f}'.format(pi)) 3.14

3.14 3.14

可以使用填充字符来扩充对齐说明符,这样将使用指定的字符而不是默认的空格来填充。

>>> "{:$^15}".format(" WIN BIG ") '$$$ WIN BIG $$$'

还有更具体的说明符=,它指定将填充字符放在符号和数字之间。

>>> print('{0:10.2f}\n{1:10.2f}'.format(pi, -pi)) 3.14

-3.14

>>> print('{0:10.2f}\n{1:=10.2f}'.format(pi, -pi)) 3.14

- 3.14

如果要给正数加上符号,可使用说明符+(将其放在对齐说明符后面),而不是默认的-。如 果将符号说明符指定为空格,会在正数前面加上空格而不是+。

>>> print('{0:-.2}\n{1:-.2}'.format(pi, -pi)) #默认设置 3.1

-3.1

>>> print('{0:+.2}\n{1:+.2}'.format(pi, -pi)) +3.1

-3.1

>>> print('{0: .2}\n{1: .2}'.format(pi, -pi)) 3.1

-3.1

需要介绍的最后一个要素是井号(#)选项,你可将其放在符号说明符和宽度之间(如果指 定了这两种设置)。这个选项将触发另一种转换方式,转换细节随类型而异。例如,对于二进制、

八进制和十六进制转换,将加上一个前缀。

——————————

① 如果要使用随区域而异的千位分隔符,应使用类型说明符n。

1

>>> "{:b}".format(42) '101010'

>>> "{:#b}".format(42) '0b101010'

对于各种十进制数,它要求必须包含小数点(对于类型g,它保留小数点后面的零)。

>>> "{:g}".format(42) '42'

>>> "{:#g}".format(42) '42.0000' width = int(input('Please enter width: ')) price_width = 10

item_width = width - price_width

header_fmt = '{{:{}}}{{:>{}}}'.format(item_width, price_width) fmt = '{{:{}}}{{:>{}.2f}}'.format(item_width, price_width) print('=' * width)

print(header_fmt.format('Item', 'Price')) print('-' * width)

print(fmt.format('Apples', 0.4)) print(fmt.format('Pears', 0.5)) print(fmt.format('Cantaloupes', 1.92))

print(fmt.format('Dried Apricots (16 oz.)', 8)) print(fmt.format('Prunes (4 lbs.)', 12)) print('=' * width)

这个程序的运行情况类似于下面这样:

Please enter width: 35

===================================