读完这一篇,字符串格式化界的“白富美”(f-strings)抱回家!
f-strings
从Python 3.6开始,新引入了一种字符串格式化方法,称为“格式化字符串常量”(formatted string literal),简称f-strings。相比于%、str.format、string.Template这些字符串格式化方法,f-strings在功能上一点不逊色,性能更优,而且更易读、更简洁,也不易出错。如果你使用的是Python3.6以上的版本,建议你使用f-strings来格式化字符串。
f-strings是以f或F打头的字符串,形如f'Ilove {name}'、F'I am {name} and I am {age} years old.'。请注意,在f(或F)与引号之间不要有空格,否则会引起语法错误。另外,对于f后面字符串的引号,你可以使用单引号、双引号或三引号,但是要配对使用,不可出现交叉,相同引号不能嵌套,这与创建普通字符串时对引号的使用要求是一样的。f-strings提供了一种把表达式嵌入字符串中的方法,而且语法相当简洁,我们只需要把表达式直接放入一对花括号中就行了。请注意,f-strings中可以有多个花括号,但这些花括号必须成对出现,不允许有单个左花括号或右花括号存在。程序运行时会计算表达式的值,并使用结果值替换表达式,最终形成一个完整的字符串。这里所说的“表达式”可以是任意表达式,包括变量、函数、各种运算表达式、条件表达式,以及lambda表达式等。
>>> name = 'Lily'
# 使用单引号
>>> f'I love {name}'
'I love Lily'
# f和'之间有空格
>>> f 'I love {name}'
SyntaxError: invalid syntax
# 使用双引号
>>> f"I love {name}"
'I love Lily'
# 使用三引号
>>> f'''I love {name}'''
'I love Lily'
>>> f'''I love {name},
and Jack loves {name},too.'''
'I love Lily,\nand Jack loves Lily,too.'
# 调用字符串的upper()函数
# 把名字转换为大写
>>> f'I love {name.upper()}'
'I love LILY'
>>> f'The area of the circle is{3.14*2*2}'
'The area of the circle is 12.56'
当f-strings引用字典中的值时,一定要特别注意引号使用的问题,相同引号不要进行嵌套。
# 单引号出现了嵌套
>>> dic = {'name':'Lily','age':18}
>>> f'I am {dic['name']},and I am{dic['age']} years old.'
SyntaxError: invalid syntax
>>> f'I am{dic["name"]},and I am {dic["age"]} years old.'
'I am Lily,and I am 18 years old.'
如果字符串本身就包含花括号,要求原样保留在最终结果中,为此需要在f-strings中使用两个花括号进行转义。连续两个左花括号代表结果字符串中一个左花括号,连续两个右括号代表结果字符串中的一个右括号。
>>> f'{{{"lily".upper()} }}'
'{ LILY }'
>>>f'{{{"lily".upper()}}}'
'{LILY}'
# 连续两个花括号之间不能有空格
>>> f'{ {{"lily".upper()} }}'
Traceback (most recent call last):
File "<pyshell#68>", line 1, in <module>
f'{ { {"lily".upper()} }}'
TypeError: unhashable type: 'set'
除了表达式之外,花括号中还可以有格式说明符,如下:
>>> f'π的值是{pi:.2f}'
'π的值是3.14'
>>> f'{75:+#010x}'
'+0x000004b'
f-strings的完整形式如下:
注:上面中的[]表示是可选的。
- 表达式(expression):对于表达式,我们应该注意如下几点:
f-strings要求花括号中必须有表达式。
f-strings可以包含任意表达式,它可以是变量、常量、各种运算表达式、条件表达式、函数等。当然,f-strings中的表达式可以是lambda表达式,但lambda表达式中的冒号(:)容易被f-strings误认为是表达式和格式说明符之间的冒号分隔符。为了避免出现这个问题,使用lambda表达式时,需要先把它放入到一个圆括号里面,然后再放入花括号中,例如:
>>> import math
>>> r = 3
>>> f'半径为{r}的圆的面积是:{(lambda r:math.pi*r*r)(r):.2f}'
'半径为3的圆的面积是:28.27'
表达式中不应该包含:或!两个符号,但这两个符号允许出现在表达式所包含的字符串之中。
表达式中也不允许出现反斜杠(\),也就是说,我们无法在表达式中使用转义字符进行转义操作(但转义字符可以用在text之中进行转义)。
以#打头的文本不允许直接出现在表达式中。但允许出现在表达式所包含的字符串之中。
表达式中允许出现!=运算符
- 类型转换标志:类型转换标志和str.format()的类型转换标志一样,有三个可能的取值,分别是!s、!r、!a。设置为!s表示调用str()对表达式进行转换(默认方式);!r表示调用repr()对表达式进行转换;!a表示调用ascii()对表达式进行转换。这些转换发生在调用format()函数使用格式说明符对表达式做格式化之前。
- 格式说明符(format specifier):f-strings的格式说明符和str.format()的格式说明符是一样的。给出了格式化说明符之后,这些说明符会被作为参数传入__format__函数对表达式的值进行格式化,若不指定格式说明符,则向__format__函数传入空字符串。有关格式说明符的内容,我们在前面讲解str.format()时已经讲过了,这里就不再赘述了。
请注意:格式化说明符中可以嵌套使用表达式。f-strings执行时会先计算这些嵌套表达式,得到完整的格式化说明符,然后再使用格式化说明符格式化外层表达式。例如:
>>> pi= 3.1415926
>>>width = 10
>>>precision = 3
>>> f'π的值是:{pi:{width}.{precision}f}'
'π的值是: 3.142'
下面举一些使用格式化说明符的例子:
>>>import math
>>>math.pi
3.141592653589793
# *指填充符号
# ^居中对齐
# +显示符号
# 10是替换字段宽度
# .3是精度
# f是浮点数
>>>f'The value of PI is {math.pi:*^+10.3f}'
'The value of PIis **+3.142**'
# =:内容右对齐,且只对数字类型有效。
# 若有符号,将其放在填充字符左侧,
# 形成“符号+填充字符+数字”的格式
>>>f'The value of PI is {math.pi:*=+10.3f}'
'The value of PIis +****3.142'
# #表示显示二进制、八进制、十六进制前缀
# 0表示空白处填充0,相当于把fill设置为0且使用=对齐方式。
# 10替换字段宽度
# X表示把十进制整数转换成十六进制数
>>>f'{55:+#010X}'
'+0X0000037'
# ,为千分位符号
>>>digit = 45724561
>>>f'The digit is {digit:+015,d}'
'The digit is+00,045,724,561'
友情提示:有关格式化说明符的详细内容,请参考公众号里介绍str.format()的文章,那里有非常详细的讲解。
“人生苦短,要学Python”
在f-strings中,f和r(或R)可以结合使用,且先后顺序任意(fr或rf),用以创建原生f-strings。上面讲解中,我们提到在f-strings的花括号外面的字符串中可以使用转义字符进行转义处理。但,若f-strings前面使用了fr或rf作为前缀,则这些转义字符会被当成普通的字符,失去转义功能,这在构建正则表达式时会非常有用。
>>> print(f'I love\n{"Lily"}')
I love
Lily
# 原生f-strings
# 转义字符被当成普通字符
# 失去转义功能
>>> print(fr'I love\n{"Lily"}')
I love\n Lily
另外,相邻的f-strings和普通字符串会自动连接在一起。两个相邻的普通字符串会在编译时连接起来,而f-strings只有运行时才进行连接。
>>> name1 = 'Lily'
>>> name2 = 'Python'
>>> 'I ''love'f'{name1}'f'{"and":^5}'f'{name2}'
'I love Lily and Python'
# 各个字符串之间可以有任意多个空格
# 连接时这些空格会被忽略
>>> 'I ' 'love ' f'{name1}' f'{"and":^5}' f'{name2}'
'I love Lily and Python'
尽管f-strings支持把整个表达式写入{}之中,但有时这样做会引起困扰,比如:
>>> f'{{1:2,3:4}}'
'{1:2,3:4}'
>>> f'{ {1:2,3:4} }'
'{1: 2, 3: 4}'
在第一个f-strings例子中,外层{}与内层{}之间没有空格,所以里面的{{和}}是用来做转义,分别表示一个左花括号({)和一个右花括号(}),这从结果字符串中可以看到这一点。
而在第二个f-strings例子中,外层{}与内层{}之间有空格,内层{}表示的是一个字典,外层{}表示对内层字典表达式求值,结果是字典本身。我们可以把第二个f-strings例子改写为:
>>> dic = {1:2,3:4}
>>> f'{dic}'
'{1: 2, 3: 4}'
有些表达式直接写在f-strings中,容易产生令人困惑的结果。对于这样的表达式,建议大家不要把整个表达式放入f-strings之中,而像上面改写的那样分开写,这可以让f-strings变得更加清晰可读。
值得注意的是,f-strings也可以用来格式化日期时间。不同于普通字符串,日期时间有自己的一套格式化方法。使用f-strings格式化日期时间时,前面提到的格式化说明符将不再起作用,必须使用日期时间特有的格式化符号,具体有哪些格式化符号,请参照下表。
格式化符号 |
含义 |
举例 |
%a |
星期几(本地缩写) |
Sun, Mon, …, Sat |
%A |
星期几(本地全名) |
Sunday, Monday, …, Saturday |
%w |
星期几(以十进制数0-6表示,0代表星期天,6代表星期六) |
0, 1, …, 6 |
%u |
星期几(ISO 8601标准,以十进制数表示,1代表星期一) |
1, 2, …, 7 |
%d |
日(以两位十进制数表示,不足两位用0补齐) |
01, 02, …, 31 |
%b |
月份(本地缩写) |
Jan, Feb, …, Dec |
%B |
月份(本地全名) |
January, February, …, December |
%m |
月份(以两位十进制数表示,不足两位用0补齐) |
01, 02, …, 12 |
%y |
年(年份后两位数,以两位十进制数表示,不足两位用0补齐) |
00, 01, …, 99 |
%Y |
年(以四位十进制数表示,不足四位用0补齐) |
0001, 0002, …, 2013, 2014, …, 9998, 9999 |
%G |
年(ISO8601标准,以四位十进制数表示,不足四位用0补齐,包含ISO大部分周(%V)) |
0001, 0002, …, 2013, 2014, …, 9998, 9999 |
%H |
时(24时制,以两位十进制数表示,不足两位用0补齐) |
00, 01, …, 23 |
%I |
时(12时制,以两位十进制数表示,不足两位用0补齐) |
01, 02, …, 12 |
%p |
上午/下午的本地表示 |
AM, PM |
%M |
分(以两位十进制数表示,不足两位用0补齐) |
00, 01, …, 59 |
%S |
秒(以两位十进制数表示,不足两位用0补齐) |
00, 01, …, 59 |
%f |
微秒(以六位十进制数表示,不足六位用0补齐) |
000000, 000001, …, 999999 |
%z |
UTC偏移(格式为±HHMM[SS[.ffffff]],若未指定时区,则返回空字符串) |
+0000, -0400, +1030, +063415, -030712.345216 |
%Z |
时区名(若未指定时区,则返回空字符串) |
UTC, EST, CST |
%j |
一年中的第几天(以三位十进制数表示,不足三位用0补齐) |
001, 002, …, 366 |
%U |
一年中的第几周(以星期天作为星期的第一天,以两位十进制数表示,不足两位用0补齐,全年第一个星期天之前的那个周称为第0周) |
00, 01, …, 53 |
%W |
一年中的第几周(以星期一作为星期的第一天,以两位十进制数表示,不足两位用0补齐,全年第一个星期一之前的那个周称为第0周) |
00, 01, …, 53 |
%V |
一年中的第几周(ISO8601标准,以星期一作为星期的第一天,以两位十进制数表示,不足两位用0补齐,包含1月4日的那一周为第01周) |
00, 01, …, 53 |
%c |
适合本地的日期时间表示 |
Tue Aug 16 21:30:00 1988 (en_US); Di 16 Aug 21:30:00 1988 (de_DE) |
%x |
适合本地的日期表示 |
08/16/88 (None); 08/16/1988 (en_US); 16.08.1988 (de_DE) |
%X |
适合本地的时间表示 |
21:30:00 (en_US); 21:30:00 (de_DE) |
%% |
百分号 |
% |
%T |
显示时分秒(格式hh:mm:ss) |
10:47:00 |
%D |
显示月日年(格式mm/dd/yy) |
01/12/19 |
%R |
显示时分(格式hh:mm) |
10:47 |
%t |
水平制表符 |
|
%n |
换行 |
|
%F |
显示年月日(格式yyyy-mm-dd) |
2019-01-12 |
>>> import datetime
>>> dt = datetime.datetime.today()
>>> f'The time is {dt:%Y-%m-%d %a%H:%M:%S}'
'The time is 2019-01-12 Sat 10:47:00'
>>> f'The time is {dt:%F %a %T}'
'The time is 2019-01-12 Sat 10:47:00'
关注我
“人生苦短,要学Python”
读完这一篇,字符串格式化界的“白富美”(f-strings)抱回家!的更多相关文章
- Python学习之路——基础篇(1)字符串格式化
字符串格式化 Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存. 百分号方式 ...
- Python_Day_5装饰器、字符串格式化、序列化、内置模块、生成器、迭代器之篇
一.装饰器 为什么要用装饰器??? 在实际的开发环境中应遵循开发封闭原则,虽然在这个原则是用的面向对象开发,但也适用于函数式编程,简单地说,它规定已经实现的功能代码不是允许修改的,但是可以被扩展: 封 ...
- Python开发【第一篇】Python基础之字符串格式化
字符串格式化 Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-310 ...
- 第二篇*2、Python字符串格式化
1.字符串格式化 Python的字符串格式化有两种方式: 百分号方式.format方式 1)百分号方式 %[(name)][flags][width].[precision]typecode (nam ...
- 第十篇 Python的字符串格式化
字符串格式化:就是按照你的意愿做一个拼接的过程. 1. 字符串格式化的第一种方式:百分号方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存. %[ ...
- Java 字符串格式化详解
Java 字符串格式化详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 文中如有纰漏,欢迎大家留言指出. 在 Java 的 String 类中,可以使用 format() 方法 ...
- C++字符串格式化库:CPPFormatLibrary
这个是很久之前写的,去年总结了一下,将其单独提取出来,作为一个开源库放到了GitHub上,然而CPPFormat之类的名字都已经被抢注了,结果只好注册了一个这么蛋疼的名字:CPPFormatLibra ...
- PHP字符串格式化特点和漏洞利用点
转载至: https://www.anquanke.com/post/id/170850 PHP中的格式化字符串函数 在PHP中存在多个字符串格式化函数,分别是printf().sprintf().v ...
- 【Python】更优的字符串格式化方式 -- "format"替代"%s"
背景 前段时间看了一篇介绍Python的代码技巧的文章,建议格式化字符串时使用"format"代替使用"%",但是没有说明原因.各博客网站介绍相关用法的博客很多 ...
随机推荐
- vxlan基础
1. 为什么需要Vxlan 普通的VLAN数量只有4096个,无法满足大规模云计算IDC的需求,而IDC为何需求那么多VLAN呢,因为目前大部分IDC内部结构主要分为两种L2,L3.L2结构里面,所有 ...
- 转载 * jQuery实现动态分割div—通过拖动分隔栏实现上下、左右动态改变左右、上下两个相邻div的大小
由jQuery实现上下.左右动态改变左右.上下两个div的大小,需要自己引入jquery1.8.0.min.js包 可用于页面布局. //============================ind ...
- ubuntu 14.04 重装机 安装笔记 无线网卡+cuda+nvidia
1. 安装QA6714 无线网卡重要参考网页 #22 回答 https://bugs.launchpad.net/ubuntu/+source/linux-firmware/+bug/1520343? ...
- Hadoop集群启动
1.初始化集群 要启动Hadoop集群,需要启动HDFS和YARN两个集群 注意:首次启动HDFS时,必须对其进行格式化操作.本质上是一些清理和准备工作, 因为此时的HDFS在物理上还是不存在的 命令 ...
- QT自定义控件系列(二) --- Loading加载动画控件
本系列主要使用Qt painter来实现一些基础控件.主要是对平时自行编写的一些自定义控件的总结. 为了简洁.低耦合,我们尽量不使用图片,qrc,ui等文件,而只使用c++的.h和.cpp文件. 由于 ...
- DNS学习笔记
一.域名的层级结构 主机名.次级域名.顶级域名.根域名 # 即 host.sld.tld.root 比如,域名math.stackexchange.com显示为math.stackexchange.c ...
- Ubuntu 将其他盘挂载到/home的子目录下
Ubuntu 14.04 将其他盘挂载到/home的子目录下当安装完Ubuntu系统,由于当时没有注意,分配的分区空间太小.经过一段时间安装了各式各样的软件后,常常会遇到/home目录下空间不够的情况 ...
- [LeetCode&Python] Problem 27. Remove Element
Given an array nums and a value val, remove all instances of that value in-placeand return the new l ...
- i3wm菜单
抛弃i3-dmenu-desktop吧,投入到 j4-demu-desktop 速度超快
- scrollview中edittext失去焦点问题
//edittext获取焦点后会瞬间失去,暂时使用这种笨方法解决(获取到焦点后过300ms再获取一次) public void requesFocus() { mEditName.setOnFocus ...