![深入理解python中[1:]、%a格式化与字节求和的用法](https://img.php.cn/upload/article/001/246/273/175790187576172.jpg)
在Python编程中,我们经常会遇到一些简洁但功能强大的代码片段。以下我们将详细剖析一个涉及标准输入处理、列表切片、字符串格式化以及字节操作的示例,帮助读者深入理解其背后的机制。
for s in [*open(i:=0)][1:]:
i += 1
print(f'Case #{i}:', sum(b'%a' % s) % 34)首先,代码中的open(0)是一个关键点。在类Unix系统中,文件描述符0通常代表标准输入(sys.stdin)。因此,open(0)的作用是打开标准输入流,使其可以被读取。
[*open(0)]这一结构则更为精妙。open(0)返回的是一个迭代器,它会逐行读取标准输入。通过在迭代器前加上星号*进行解包(unpacking),Python会将标准输入的所有行读取完毕,并将每一行作为一个字符串元素,收集到一个新的列表中。例如,如果输入是:
Line 1 Line 2 Line 3
那么[*open(0)]将生成一个列表:['Line 1 ', 'Line 2 ', 'Line 3 ']。
立即学习“Python免费学习笔记(深入)”;
此外,i:=0是Python 3.8引入的“海象运算符”(walrus operator)的一个应用。它允许在表达式内部进行变量赋值。在这里,它在open(0)被调用时将变量i初始化为0,这是一种代码高尔夫(code golfing)式的写法,旨在减少代码行数。
在获取到包含所有输入行的列表后,代码紧接着使用了[1:]进行列表切片操作。
列表切片是Python中处理序列(如列表、字符串、元组)的常用方法。其基本语法是sequence[start:end:step]。
[1:]的含义是从索引为1的元素开始,直到列表的末尾。这意味着原始列表中的第一个元素(索引为0的元素)将被跳过。在上述示例中,它用于忽略标准输入的第一行。
因此,[*open(i:=0)][1:]的整体效果是:读取标准输入的所有行,将其转换为一个列表,然后返回该列表除了第一行之外的所有行,供for循环迭代。
循环体内部的核心计算是sum(b'%a' % s) % 34。我们首先聚焦于b'%a' % s。
%a是Python字符串格式化(printf-style string formatting)中的一个特殊格式化符。它用于将对象转换为其ASCII表示。具体来说,它会返回一个字符串,其中包含对象的打印表示,但会使用x、u或U等转义序列来表示非ASCII字符。这与Python 2中repr()函数的行为类似。
示例:
>>> '%a' % 'foobar' "'foobar'" >>> '%a' % '你好' "'\u4f60\u597d'"
可以看到,它不仅将字符串内容包含在引号中,还会对非ASCII字符进行转义。
在'%a' % s的结果前加上b前缀,如b'...',表示这是一个字节字面量(bytes literal)。Python中的bytes对象是不可变的字节序列,与字符串(str)对象不同,bytes对象存储的是原始字节数据,而不是Unicode字符。
当我们将'%a' % s的结果(一个字符串)与b''结合时,实际上是将该字符串编码成字节序列。默认情况下,Python 3的字符串会以UTF-8编码。然而,b'%a' % s这种写法,其内部机制是先将s格式化为字符串,然后将这个字符串表示的字面值作为字节序列来处理。
示例:
>>> s = 'foobar' >>> formatted_s = '%a' % s >>> formatted_s "'foobar'" >>> b_formatted_s = b'%a' % s # 相当于 b"'foobar'" >>> b_formatted_s b"'foobar'"
这里,b'%a' % s的结果是一个bytes对象,其内容是字符串"'foobar'"的ASCII字节表示。
获取到b'%a' % s生成的bytes对象后,下一步是sum(...)。
在Python中,bytes对象本质上是整数序列(每个字节的值范围是0-255)。对一个bytes对象调用sum()函数,会将其内部所有字节的整数值累加起来。
示例:
>>> b_obj = b'abc' # 'a'的ASCII值是97, 'b'是98, 'c'是99
>>> sum(b_obj)
294 # 97 + 98 + 99 = 294
>>> sum(b'%a' % 'foobar') # 对应 b"'foobar'"
>>> sum(b"'foobar'")
711 # 39(') + 102(f) + 111(o) + 111(o) + 98(b) + 97(a) + 114(r) + 39(') = 711最后,% 34是对求和结果进行模运算。模运算返回除法的余数。在这里,它将字节值之和除以34,并返回其余数。这个操作通常用于将一个大范围的数值映射到一个较小的、循环的范围内。
示例:
>>> sum(b'%a' % 'foobar') % 34 31 # 711 % 34 = 31
现在,我们可以将所有部分整合起来,理解整个代码的工作流程:
假设输入:
Header line (will be skipped) Apple Banana Cherry
执行过程:
通过对这段代码的深入剖析,我们不仅理解了各个组件的功能,也掌握了Python中处理标准输入、列表切片、字符串到字节转换以及字节序列操作的高级技巧。
以上就是深入理解Python中[1:]、%a格式化与字节求和的用法的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号