全网最适合入门的面向对象编程教程:54 Python字符串与序列化-字符串格式化与format方法

news/2025/1/12 15:56:13/文章来源:https://www.cnblogs.com/FreakEmbedded/p/18438704

全网最适合入门的面向对象编程教程:54 Python 字符串与序列化-字符串格式化与 format 方法

image

摘要:

在 Python 中,字符串格式化是将变量插入到字符串中的一种方式,Python 提供了多种字符串格式化的方法,包括旧式的 % 格式化、新式的 str.format 方法以及 f-string(格式化字符串字面量)。

原文链接:

FreakStudio的博客

往期推荐:

学嵌入式的你,还不会面向对象??!

全网最适合入门的面向对象编程教程:00 面向对象设计方法导论

全网最适合入门的面向对象编程教程:01 面向对象编程的基本概念

全网最适合入门的面向对象编程教程:02 类和对象的 Python 实现-使用 Python 创建类

全网最适合入门的面向对象编程教程:03 类和对象的 Python 实现-为自定义类添加属性

全网最适合入门的面向对象编程教程:04 类和对象的Python实现-为自定义类添加方法

全网最适合入门的面向对象编程教程:05 类和对象的Python实现-PyCharm代码标签

全网最适合入门的面向对象编程教程:06 类和对象的Python实现-自定义类的数据封装

全网最适合入门的面向对象编程教程:07 类和对象的Python实现-类型注解

全网最适合入门的面向对象编程教程:08 类和对象的Python实现-@property装饰器

全网最适合入门的面向对象编程教程:09 类和对象的Python实现-类之间的关系

全网最适合入门的面向对象编程教程:10 类和对象的Python实现-类的继承和里氏替换原则

全网最适合入门的面向对象编程教程:11 类和对象的Python实现-子类调用父类方法

全网最适合入门的面向对象编程教程:12 类和对象的Python实现-Python使用logging模块输出程序运行日志

全网最适合入门的面向对象编程教程:13 类和对象的Python实现-可视化阅读代码神器Sourcetrail的安装使用

全网最适合入门的面向对象编程教程:全网最适合入门的面向对象编程教程:14 类和对象的Python实现-类的静态方法和类方法

全网最适合入门的面向对象编程教程:15 类和对象的 Python 实现-__slots__魔法方法

全网最适合入门的面向对象编程教程:16 类和对象的Python实现-多态、方法重写与开闭原则

全网最适合入门的面向对象编程教程:17 类和对象的Python实现-鸭子类型与“file-like object“

全网最适合入门的面向对象编程教程:18 类和对象的Python实现-多重继承与PyQtGraph串口数据绘制曲线图

全网最适合入门的面向对象编程教程:19 类和对象的 Python 实现-使用 PyCharm 自动生成文件注释和函数注释

全网最适合入门的面向对象编程教程:20 类和对象的Python实现-组合关系的实现与CSV文件保存

全网最适合入门的面向对象编程教程:21 类和对象的Python实现-多文件的组织:模块module和包package

全网最适合入门的面向对象编程教程:22 类和对象的Python实现-异常和语法错误

全网最适合入门的面向对象编程教程:23 类和对象的Python实现-抛出异常

全网最适合入门的面向对象编程教程:24 类和对象的Python实现-异常的捕获与处理

全网最适合入门的面向对象编程教程:25 类和对象的Python实现-Python判断输入数据类型

全网最适合入门的面向对象编程教程:26 类和对象的Python实现-上下文管理器和with语句

全网最适合入门的面向对象编程教程:27 类和对象的Python实现-Python中异常层级与自定义异常类的实现

全网最适合入门的面向对象编程教程:28 类和对象的Python实现-Python编程原则、哲学和规范大汇总

全网最适合入门的面向对象编程教程:29 类和对象的Python实现-断言与防御性编程和help函数的使用

全网最适合入门的面向对象编程教程:30 Python的内置数据类型-object根类

全网最适合入门的面向对象编程教程:31 Python的内置数据类型-对象Object和类型Type

全网最适合入门的面向对象编程教程:32 Python的内置数据类型-类Class和实例Instance

全网最适合入门的面向对象编程教程:33 Python的内置数据类型-对象Object和类型Type的关系

全网最适合入门的面向对象编程教程:34 Python的内置数据类型-Python常用复合数据类型:元组和命名元组

全网最适合入门的面向对象编程教程:35 Python的内置数据类型-文档字符串和__doc__属性

全网最适合入门的面向对象编程教程:36 Python的内置数据类型-字典

全网最适合入门的面向对象编程教程:37 Python常用复合数据类型-列表和列表推导式

全网最适合入门的面向对象编程教程:38 Python常用复合数据类型-使用列表实现堆栈、队列和双端队列

全网最适合入门的面向对象编程教程:39 Python常用复合数据类型-集合

全网最适合入门的面向对象编程教程:40 Python常用复合数据类型-枚举和enum模块的使用

全网最适合入门的面向对象编程教程:41 Python常用复合数据类型-队列(FIFO、LIFO、优先级队列、双端队列和环形队列)

全网最适合入门的面向对象编程教程:42 Python常用复合数据类型-collections容器数据类型

全网最适合入门的面向对象编程教程:43 Python常用复合数据类型-扩展内置数据类型

全网最适合入门的面向对象编程教程:44 Python内置函数与魔法方法-重写内置类型的魔法方法

全网最适合入门的面向对象编程教程:45 Python实现常见数据结构-链表、树、哈希表、图和堆

全网最适合入门的面向对象编程教程:46 Python函数方法与接口-函数与事件驱动框架

全网最适合入门的面向对象编程教程:47 Python函数方法与接口-回调函数Callback

全网最适合入门的面向对象编程教程:48 Python函数方法与接口-位置参数、默认参数、可变参数和关键字参数

全网最适合入门的面向对象编程教程:49 Python函数方法与接口-函数与方法的区别和lamda匿名函数

全网最适合入门的面向对象编程教程:50 Python函数方法与接口-接口和抽象基类

全网最适合入门的面向对象编程教程:51 Python函数方法与接口-使用Zope实现接口

全网最适合入门的面向对象编程教程:52 Python函数方法与接口-Protocol协议与接口

全网最适合入门的面向对象编程教程:53 Python字符串与序列化-字符串与字符编码

更多精彩内容可看:

给你的 Python 加加速:一文速通 Python 并行计算

一文搞懂 CM3 单片机调试原理

肝了半个月,嵌入式技术栈大汇总出炉

电子计算机类比赛的“武林秘籍”

一个MicroPython的开源项目集锦:awesome-micropython,包含各个方面的Micropython工具库

Avnet ZUBoard 1CG开发板—深度学习新选择

SenseCraft 部署模型到Grove Vision AI V2图像处理模块

文档和代码获取:

可访问如下链接进行对文档下载:

https://github.com/leezisheng/Doc

image

本文档主要介绍如何使用 Python 进行面向对象编程,需要读者对 Python 语法和单片机开发具有基本了解。相比其他讲解 Python 面向对象编程的博客或书籍而言,本文档更加详细、侧重于嵌入式上位机应用,以上位机和下位机的常见串口数据收发、数据处理、动态图绘制等为应用实例,同时使用 Sourcetrail 代码软件对代码进行可视化阅读便于读者理解。

相关示例代码获取链接如下:https://github.com/leezisheng/Python-OOP-Demo

正文

字符串格式化是将变量值插入到字符串中的占位符位置的过程。这使得能够创建动态的文本,其中一些部分可能需要根据不同情况进行替换。

在 Python 中字符串格式化的方式包括 Print 函数输出格式化和 Format 函数格式化。

使用 Print 函数格式化符号实现格式化

常用格式化符号包括:

  • (1)%c:格式化字符及其 ASCII 码;
  • (2)%s:格式化字符串;
  • (3)%d:格式化整数;
  • (4)%u:格式化无符号整型;
  • (5)%o:格式化无符号八进制数;
  • (6)%x:格式化无符号十六进制数;
  • (7)%f:格式化浮点数字,可指定小数点后的精度;
  • (8)%e:用科学计数法格式化浮点数;
  • (9)%g:%d 和 %e 的简写。

示例代码如下所示:

print("%c" % 'a')
print("%s" % "string")
print("%s" % 123)   
print("%d" % 100.0)

运行结果如下:

image

同时转换符格式化 (conversion specifier) 可以引用字典变量。

转换符的格式为 %(mapping_key)flags,mapping_key 指明引用变量的名称,flags 指明转换格式。

示例代码如下:

print('%(language)s has %(number)01d quote types.'% {'language': "Python", "number": 2})

运行结果如下:

image

使用 format()方法进行格式化

内置的字符串类提供了通过使用 PEP 3101 所描述的 format()方法进行复杂变量替换和值格式化的能力。string 模块中的 Formatter 类允许你使用与内置 format()方法相同的实现来创建并定制你自己的字符串格式化行为。

image

str.format()方法和 Formatter 类共享相同的格式字符串语法(虽然对于 Formatter 类来说,其子类可以定义它们自己的格式字符串语法)。任何字符串都可以通过调用 format()方法而编写一个格式化字符串。这个方法返回一个新的字符串,其中的特殊字符将会替换成传入该方法的参数以及关键字参数。format 方法不限定参数数量,它使用我们前面提到的方法传参中*args 和**kwargs 语法。

image

在格式化字符串中被替换的特殊符号是开闭花括号:{和}。我们可以成对地插入,最终其会按照顺序被 str.format 方法中传入的位置参数所替换。

示例代码如下:

template = "Hello {}, you are currently {}." 
print(template.format('Dusty', 'writing'))

运行结果如下:

image

str.format()方法可以使用位置参数或关键字参数来填充占位符。位置参数是按顺序传递的,而关键字参数使用占位符名称来匹配值。

示例代码如下,运行结果同上。

template = "Hello {0}, you are currently {1}."
print(template.format('Dusty', 'writing'))
template = "Hello {name}, you are currently {doing}."
print(template.format(name='Dusty', doing='writing'))

我们不是只能传递字符串变量给 format 方法,任何基本类型,例如可以被打印出来的整数或浮点数都可以。更有趣的是,复杂对象,包括列表、元组、字典以及任意对象都可以使用;在 format 的字符串中我们可以通过索引和对象的属性(方法不行)访问变量。例如在下面的例子中,我们输出一个邮件消息,我们在邮箱地址中将发件人和收件人组合成元组,并将主题和消息存放在字典中,就可以这样来格式化:

emails = ("a@example.com", "b@example.com")
message = {'subject': "You Have Mail!",'message': "Here's some mail for you!"}
template = """ 
From: <{0[0]}> 
To: <{0[1]}> 
Subject: {message[subject]} 
{message[message]}"""
print(template.format(emails, message=message))

我们在传递 emails 元组时传递了一个基于位置的参数,这两个邮箱地址可以通过 0[x]获得,其中 x 可以是 0 或 1,表示元组中第一个或第二个元素。第一个 0 表示传入 format 的第一个位置参数(在这个例子中是 emails 元组)。

我们在传递 message 字典时传递了一个基于关键字的参数,其中通过字符串键名访问字典时,使用 message[subject]和 message[message],注意这里和平常访问字典中值时有所不同,我们不需要加引号。

print(message['subject'])

如果有嵌套的数据结构,我们甚至可以实现多层查询。这里建议不要这样做,因为这样一来,模板字符串很快就会变得难以理解。如果我们有一个字典包含了元组,可以这样做,代码运行结果同上:

emails = ("a@example.com", "b@example.com")
message = {'emails': emails,'subject': "You Have Mail!",'message': "Here's some mail for you!"}
template = """ 
From: <{0[emails][0]}> 
To: <{0[emails][1]}> 
Subject: {0[subject]} 
{0[message]}"""
print(template.format(message))

实际上,我们也可以给 format 函数传递任意对象作为参数,然后用点号标记访问对象的属性。在如下代码中,我们使用 format 函数输出了类的名字、描述和属性/方法。

class SensorClass(SerialClass):'''传感器类,继承自SerialClass'''... ...
if __name__ == "__main__":template = '''Class Name : <{0.__name__}>Class Description :  <{0.__doc__}>Class Method and Class Properties : <{0.__dict__}>    '''print(template.format(SensorClass))

运行结果如下所示:

image

通常来说,我们会在想要格式化的对象已经存在的情况下使用这种查询方式,但不会为了在模板中使用对象而创建一个类。

在模板字符串中引入变量确实极具便利性,然而,有时候为了确保输出结果的准确性,我们需要对这些变量进行适当的调整。例如,在进行准确率计算时,我们可能会得到冗长的小数,但在特定的应用场景中,我们并不希望这些详尽的小数点后的数值出现在最终的展示中。

此时,str.format()方法就显得尤为实用。它不仅允许我们灵活地控制输出的格式,还提供了诸多选项以满足不同的格式化需求,如指定小数位数、调整文本对齐方式以及设置填充字符等。通过这些功能,我们可以更加精准地控制输出内容的呈现方式,从而确保信息的传达既准确又符合预期的展示效果。

标准格式说明符的一般形式如下:

format_spec     ::=  [[fill]align][sign]["z"]["#"]["0"][width][grouping_option]["." precision][type]
fill            ::=  <any character>
align           ::=  "<" | ">" | "=" | "^"
sign            ::=  "+" | "-" | " "
width           ::=  digit+
grouping_option ::=  "_" | ","
precision       ::=  digit+
type            ::=  "b" | "c" | "d" | "e" | "E" | "f" | "F" | "g" | "G" | "n" | "o" | "s" | "x" | "X" | "%"

image

image

image

这里,type 参数具体含义如下:

(1) s: string, 字符串;
(2) d: decimal integer, 十进制数;
(3) i: integer, 用法同%d;
(4) u: unsigned integer, 无符号十进制数;
(5) f: float, 浮点数(默认保留小数点后6位);
(6) F: Float, 浮点数(默认保留小数点后6位);
(7) e: exponent, 将数字表示为科学计数法(小写e, 默认保留小数点后6位);
(8) E: Exponent, 将数字表示为科学计数法(大写E, 默认保留小数点后6位);
(9) o: octal, 八进制数(即0-7);
(10) x: hexdecimal, 十六进制数(即0-9a-f);
(11) X: Hexdecimal, 十六进进制数(0-9A-F);
(12) g: general format, 通用格式,详见如下...;
(13) G: General format, 通用格式,详见如下...;
(14) %c: character, 将十进制数转换为所对应的unicode值;
(15) %r: representation, 调用__repr__魔法方法输出;
(16) %%: 转义%,输出百分号。

具体可看下面示例:

price = 49.95
formatted_price = "The price is {:.2f} dollars.".format(price)
print(formatted_price)

运行结果如下:

image

冒号之后的 0.2f 格式指示符表明,从左向右,对于小于 1 的值,确保小数点左侧有个 0;小数点之后保留两位数字,将输入值格式化为浮点数。

我们也可以通过占位值让每个数字占据特定数量的字符位置。这对于输出表格数据很有用,例如:

orders = [('burger', 2, 5),('fries', 3.5, 1),('cola', 1.75, 3)]
print("PRODUCT    QUANTITY    PRICE   SUBTOTAL")
for product, price, quantity in orders:subtotal = price * quantityprint("{0:10s}{1: ^9d} ${2: <8.2f}${3: >7.2f}".format(product, quantity, price, subtotal))

这里,quantity 变量的格式化操作符{1: ^9d}为例,d 表示这是一个整数值,数字 9 说明这个值需要占据 9 个字符。

但是对于整数来说,默认是用 0 而不是空格来填充的。所以我们在冒号之后添加一个空格作为占位符。插入符号说明数字按照居中方式对齐。运行结果如下:

image

我们也可以使用 < 或 > 选择不同的对齐方式(左对齐/右对齐)。这里,我们对 price 和 subtotal 变量使用相似的指示符。对于 price,我们使用{2: <8.2f};对于 subtotal,我们使用{3: >7.2f}。我们都指定了一个空格作为填充字符,不过用 < 和 > 符号分别说明数字按照长度为 8 和 7 的位置居左和居右对齐。而且,每个浮点数都保留两位小数。

对于不同的数据类型,“类型”字符的不同会导致输出格式的变化。我们已经了解了 s、d 和 f 这三种类型,它们分别用于表示字符串、整数和浮点数。实际上,大部分其他的格式指示符都可以看作是这三种基本类型的变体或扩展。例如,o 类型用于表示八进制整数,而 X 类型则用于表示十六进制整数。此外,n 类型指示符是一个特殊的存在,它允许我们根据本地的习惯对整数进行分隔。这对于浮点数,% 类型指示符则具有特殊的功能,它可以将浮点数乘以 100,从而将其转换为百分数形式。

示例代码如下:

_# 使用千位分隔符_
number = 1234567
formatted_number = "Formatted number: {:,}".format(number)
print(formatted_number)
_# 使用百分比格式_
percentage = 0.25
formatted_percentage = "Formatted percentage: {:.2%}".format(percentage)
print(formatted_percentage)

运行结果如下:

image

值得注意的是,这些标准的格式化操作符不仅可以应用于内置的数据类型,还可以应用于其他对象。对于非标准对象,我们可以定义自己的格式指示符来满足特定的需求。例如,如果我们将 datetime 对象传递给 format,就可以使用那些可用于 datetime.strftime 函数的指示符,例如:

from datetime import datetimenow = datetime.now()
_# 格式化日期和时间_
formatted_date = "Current date and time: {:%Y-%m-%d %H:%M:%S}".format(now)
print(formatted_date)

输出结果如下:

image

需要注意的是:这些指示符必须按照正确的顺序,不过它们都是可选的:首先是填充字符,其次是对齐方式,然后是大小,最后是类型。

::=  [[fill]align][sign]["z"]["#"]["0"][width][grouping_option]["." precision][type]

除了使用标准的格式化操作符,我们也可以通过重写 format 特殊方法来自定义我们自己对象的格式化操作符,可以通过查看如下文档进一步了解更加细节的操作:https://peps.python.org/pep-3101/

image

image

同时,开闭花括号:{和}除了在格式化字符串中作为被替换的特殊符号,花括号符号本身也常用于字符串。我们需要有办法跳过格式化,只作为花括号符号本身出现,而不是被替换。我们可以通过重复两次花括号来实现,例如,我们可以用 Python 来格式化一个基本的 Java 程序:

template = """
public class {0} {{public static void main(String[] args) {{System.out.println("{1}");}}
}}"""
print(template.format("MyClass", "print('hello world')"))

运行结果如下:

image

可以看到输出的类名和内容已经被两个参数所替换,而双花括号被替换为单花括号,结果得到的就是一段合法的 Java 代码。只要在模板中看到{{或}},也就是用于封闭 Java 类和方法定义的符号。我们知道 format 方法会将它们替换为单个的花括号符号,而不是替换为传入 format 方法的参数。

这里,我们也总结了常用的一些格式化标准符号使用方法:

image

总的来说,格式化操作符是 Python 中一个非常强大的工具,它允许我们根据需要灵活地控制数据的输出格式。通过深入了解这些操作符的用法和特点,我们可以编写出更加清晰、易读和高效的代码。

image

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/805469.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

智慧工地AI视频分析系统

智慧工地AI视频分析系统通过图像识别技术,智慧工地AI视频分析系统配合现场已有摄像头,不需人为干预自动识别现场作业人员穿戴是否合规如安全帽反光衣有无穿戴、高空作业是否穿戴安全带、抽烟打电话识别、人员打架、危险区域人员闯入识别、工作时间睡岗离岗识别、工地车辆识别…

员工工作服穿戴识别系统

员工工作服穿戴识别系统基于YOLO网络模型图像识别技术,员工工作服穿戴识别系统通过现场已有的监控摄像头,不需新增硬件对现场未按要求穿戴工服的违规行为实时预警,将违规行为信息及时推送给后台管理人员。员工工作服穿戴识别系统通过AI技术手段提高现场对施工作业人员穿戴监…

劳保防护用品穿戴检测系统

劳保防护用品穿戴检测系统通过Opencv深度学习技术,劳保防护用品穿戴检测系统对现场作业人员行为以及安全作业防护穿戴用品进行全天候检测,当劳保防护用品穿戴检测系统检测到现场施工人员未按照要求进行施工穿戴防护用品,劳保防护用品穿戴检测系统立即对现场违规穿戴人员或者…

加油站ai系统视频监测

加油站ai系统视频监测通过深度学习边缘计算技术,加油站ai系统视频监测对现场画面中人员作业行为实时进行检测分析,加油站ai系统视频监测不需人为干预通过AI技术识别异常违规行为信息,加油站ai系统视频监测能将风险及时发现并进行预警,提升加油站作业全流程安全系数。加油站…

办公室人员离岗识别检测系统

办公室人员离岗识别检测系统根据计算机视觉深度学习技术,办公室人员离岗识别检测系统能够7*24小时全天候自动识别工作时间监控画面中人员是否在岗位工作。办公室人员离岗识别检测系统发现监控画面中人员在工作时间没有在岗位时,不需人为干预办公室人员离岗识别检测系统会立刻…

智慧工地安全着装识别系统

智慧工地安全着装识别系统通过AI视频分析技术,智慧工地安全着装识别系统对工地现场物体的不安全状态以及施工人员的不安全行为(不按要求着装)进行自动实时分析,发现异常违规信息立即抓拍预警同步回传工地后台大数据平台提醒后台执勤人员及时处理,避免发生更危险的情况。智…

煤矿风险监测预警系统

煤矿风险监测预警系统基于YOLO网络模型视觉分析,煤矿风险监测预警系统7*24小时不间断自动识别现场人员作业行为、着装合规情况以及传送皮带撕裂跑偏等风险异常情况,煤矿风险监测预警系统检测出人员未按照要求穿安全帽反光衣、抽烟玩手机、皮带跑偏撕裂堆煤异物后,煤矿风险监…

安全帽反光背心穿戴识别系统 反光衣穿戴检测系统

安全帽反光背心穿戴识别系统 反光衣穿戴检测系统利用现场已有摄像头,安全帽反光背心穿戴识别系统 反光衣穿戴检测系统实时分析施工现场人员着装穿戴及作业行为。当安全帽反光背心穿戴识别系统 反光衣穿戴检测系统识别到现场人员未按照要求穿戴安全帽以及反光背心(又称反光衣)…

论信息显示对我生活的影响

论信息显示对我生活的影响 引子 灵感来源是昨天去"桌游社"打万智牌 他们好热情啊,直接就给我开了四包新卡,怪不好意思的 不过我那天手气不错,开出了"兄弟反目"(50元),嘻嘻:happy: 看不到信息的时候 玩《空洞骑士》的时候,像是攻击距离,攻击伤害,闪避…

《Python 基础篇》三:流程控制

Python 中的流程控制语句。Author: ACatSmiling Since: 2024-09-27Python 代码在执行时是按照自上向下顺序执行的。通过流程控制语句,可以改变程序的执行顺序,也可以让指定的程序反复执行多次。 流程控制语句分成两大类:条件判断语句,循环语句。 条件判断语句 if 语法: if…

手机为什么能够打电话上网,一文带你搞懂其原理

我们的手机为什么能够打电话能够上网,这个问题相信 很多人都思考过,为什么在千里之外,他可以借到我拨出的电话,我说话的声音可以清晰地传到他的耳中,在这小小的手机里,我们可以浏览天下所有的资讯。这究竟是怎么样做到的? 这其实是得益于蜂窝移动通信网络的发展。今天我…

[友链] 挚友之链

Mystery0の小站https://blog.mystery0.vip/本文作者:千千寰宇本文链接:https://www.cnblogs.com/johnnyzen关于博文:评论和私信会在第一时间回复,或直接私信我。版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!日常交流:大数据与软件开…