[转] 三套2022年Python十级试题以及答案解析
原文: https://www.dongwm.com/post/python-ten-level-exam/
题目 1
先看全国卷 A 的 10 道题目。
这个题目来自 Raymond Hettinger 的 Tweet 。
答案是 B. 因为- 1
(中间有空格) 其实就是-1
,也就是说可以这么表示score -= (-1)
。
题目 2
送分题,答案是 A,也就是抛 SyntaxError 错误,因为海象操作符需要使用括号不能直接用,因为需要和普通的赋值区分开来。
题目 3
题目来源:https://github.com/satwikkansal/wtfpython#-deleting-a-list-item-while-iterating
这个题目主要是考验对迭代的理解。在循环时先迭代了第一个元素 1 (索引 0) 然后 remove 删除这个元素,剩下了三个元素 2,3,4,但是注意,这里 2 的索引是 0,3 的索引是 1。下一次迭代应该是索引 1,就是迭代并删掉 3,把 2 给略过了,接着会把 4 略过。略过的就会留下,所以结果是[2, 4]
。
题目 4
送分题,答案是 D,因为min
是自带的函数,如果把它替换成其他的对象就不能正常运行了,那么就会抛错 TypeError。
题目 5
题目来源:https://github.com/satwikkansal/wtfpython#-be-careful-with-chained-operations
答案是 A,这个特别反直觉对吧。但要注意比较方式是按顺序把相邻的 2 个分别比较, 官网这么说 :
if a, b, c, …, y, z are expressions and op1, op2, …, opN are comparison operators, then a op1 b op2 c ... y opN z is equivalent to a op1 b and b op2 c and ... y opN z, except that each expression is evaluated at most once.
所以False == False in [False]
的意思是(False == False) and (False in [False])
,所以结果是 True。
题目 6
送分题,答案是 A,因为 bool 值也是一种数字 (True 为 1,False 为 0):
|
题目 7
答案是 B。这道题我就是想让大家知道判断可以直接在 print 里面写,而不需要这样:
|
题目 8
送分题,答案是 D,知识点是列表解包 (Unpacking)。
题目 9
题目来源:https://github.com/satwikkansal/wtfpython#-hash-brownies
答案是 C. 在 Python 的字典中,它不关心键的类型,只要它们的值一样那么就是同一个键值对,后面的赋值会替换前面的值:
|
题目 10
答案是 A,来源找不到了,我之前还专门写过一篇文章讲这个 一段迷惑的使用海象操作符的代码
题目 11
再看全国卷 B 的 10 道题目。
这个不了解的比较难,答案是 A。这是 Python freeze 自动创建的模块,除此之外还有__phello__
:
|
而其他选项中,__builtin__
是 Python 2 时代的模块,还有个 Python2/3 都可以用的 builtins 模块,但是没有__builtins__
。另外有__future__
和futures
但没有__futures__
,都是用来混淆的。
题目 12
题目来源:https://github.com/satwikkansal/wtfpython#-needles-in-a-haystack-
答案是 C。这个和题目 10 其实很像。你可以把它理解成这是一个赋值语句,逗号前面的赋值给 x,后面的赋值给 y。如果加上括号就是另外一个意思了:
|
这样就表示根据判断条件赋值不同的元组了。
题目 13
题目来源:https://github.com/satwikkansal/wtfpython#-the-disappearing-variable-from-outer-scope
答案是 D。而在 Python2 中,e 的结果是Exception()
,注意这个和 wtfpython 项目里的说明不符。
在 Python3 为什么直接抛 NameError 呢?因为:
|
在 except 作用域里面实际上相当于:
|
也就是说,最终在离开作用域时会把 as 的别名 N 删掉,这样 e 就不存在了,所以是 NameError。
题目 14
送分题,答案是 B。这是海象操作符的常见应用场景,首先先在(x := [1, 2])
里给 x 赋值为[1, 2]
,然后再对这个 x 执行x.extend(x)
。
题目 15
答案是 A。这个题目是展示在循环时还能做其他很多事情,例如顺便对迭代对象赋值。很久前我还发过类似的 豆瓣广播 :
题目 16
题目来源:https://github.com/satwikkansal/wtfpython#-all-sorted-
答案是 B。这个第一次我也想错了。它的问题在于对一个迭代对象做两次迭代时,后面那次的迭代开始时迭代对象已经为空了:
|
题目 17
题目来源:https://github.com/satwikkansal/wtfpython#-same-operands-different-story
送分题,答案是 A。a += [4, 5, 6]
会生成新的列表 a,但是 b 引用的是旧的 a,所以不会受到影响。
题目 18
题目来源:https://github.com/satwikkansal/wtfpython#-loop-variables-leaking-out
送分题,答案是 C。for 循环会影响作用域之外的变量值,但是有一点需要注意,从 Python 3 开始,列表解析不会影响作用域之外的变量值,举个例子:
|
题目 19
绝对的送分题,答案是 C。不过这个题目我没写好,原题目对于 Python 熟悉的开发者自动会去掉 A/B2 个答案,应该选项是:
|
这样就更具备迷惑性了。其实就是把一个 range 类型的可迭代对象在集合里面解包。
题目 20
题目来源:https://github.com/satwikkansal/wtfpython#-yielding-from-return-
这个其实是语言设计的问题,答案 C。
首先说yield from
其实就是:
|
的意思。主要考验大家对于生成器和 Python 3.3 新加入的yield from
的熟悉程度。如果一个函数内有yield
或者yield from
,那么这就是一个生成器:
|
当时设计时 生成器内可以使用 return ,事实上是停止生成器的用途,如官方所说:
"... return expr in a generator causes StopIteration(expr) to be raised upon exit from the generator."
所以return ['wtf']
等于raise StopIteration(['wtf'])
,在执行list()
的时候,就会捕捉错误直接结束。
但是我们这个例子中,直接符合的 return 的条件,就造成它直接返回了空列表 (因为一上来就 raise 了 StopIteration)。
题目 21
最后看全国卷 C 的 10 道题目。这也是我认为最难的一份卷子了。
考验对__future__
模块的了解程度,答案是 C。大家应该了解每个选项的意义,我这里就不挨个提了。
题目 22
题目来源:https://github.com/satwikkansal/wtfpython#-name-resolution-ignoring-class-scope
答案是 A。首先,嵌套在类定义中的范围会忽略在类级别绑定的变量。
另外如题目 18 里面提到的,Python 3 的列表解析 / 生成器有自己的作用域,而不会影响外面。
题目 23
题目来源:https://github.com/satwikkansal/wtfpython#-evaluation-time-discrepancy
答案是 C。在生成器表达式中,in
是声明时计算,条件判断时在运行时执行计算。所以这个题目中gen = (x for x in array if array.count(x) > 0)
也就是gen = (x for x in [1, 8, 15] if [2, 8, 22].count(x) > 0)
这个地方需要大家仔细理解。
题目 24
题目灵感:https://github.com/satwikkansal/wtfpython#-deep-down-were-all-the-same
我们先思考==
和is
的关系:==
表示值相等,is
表示指向的内容地址一样,所以对比的 2 个选项可能==
但是可能不is
,因为is
需要更高的要求,同样的,如果is
了,那么肯定==
。
我们挨个去确定。WTF() == WTF()
表示 2 个实例,肯定不会是一样的;WTF() is WTF()
既然都不相等,所以is
就更不可能了。
如果你理解==
和is
的关系,现在就可以知道答案是 C。但是为什么id(WTF()) == id(WTF())
而id(WTF()) is id(WTF())
不对呢?
在一个表达式中,如果执行 2 遍 id 函数,后面那次会被分配到相同内存位置,所以相等==
。但是id(WTF()) is id(WTF())
其实和本题目关系不大,是我发挥的选项,举个更直接的例子:
|
这个是池化的问题,如果数字不在 - 5 到 256 之间,Python 不会缓存数字对象,而 id 函数执行的结果远大于这个值所以就是 False 了。
我们在最新的 Python3.10 试一下,默认的输出不符合预期,我再换个思路:
|
题目 25
题目来源:https://github.com/satwikkansal/wtfpython#-all-true-ation-
这个其实仔细仔细分析是可以找到答案的,答案为 B。
all
的意思是把参数做循环,每个元素都符合要求才是 True,只要有一个不符合就是 False。所以all([])
是空列表结果为空,all([[]])
表示列表只有一个元素[]
,而空列表是 False,所以结果是 False。
最后 2 个选项稍微让人迷惑,列表只有一项元素,分别是[[]]
和[[[]]]
,它们都是非空的,所以布尔值是 True:
|
题目 26
这个相对比较简单,资深的 Python 开发应该会写过这样的代码,答案是 C。
在 Python 中,如果对变量赋值,那么这个变量就会编程当前范围的本地变量,所以在函数some_func
里面a += 1
,a
就成了函数里的本地变量,但是在函数范围里面,前面并没有定义 a 或者对 a 赋值,只有a += 1
,所以就抛 UnboundLocalError 了。
如果想让程序不报错,解决办法是在函数内加 global 关键字:
|
但是注意,global 要谨慎使用,如它的名字所提示的,使用它会影响全局变量的结果。
题目 27
答案是 B。这个题目考查对于字符串的strip
、rstrip
等方法的含义。默认情况下它们是用作去掉字符串行位空格的:
|
但是也可以传入其他字符串,实现replace
函数所做把对应匹配项替换成空:
|
所以原字符串中包含的.US.TXT
都会被替换,也就是. U S T X
这几个字符会被替换成空。
注意,如果只是想要移除后缀,可以使用 Python 3.9 新加入的removesuffix
/removeprefix
:
|
题目 28
题目来源:https://github.com/satwikkansal/wtfpython#-how-not-to-use-is-operator
答案是 A。这其实是 Python3.7 的一个 BUG,已经在https://bugs.python.org/issue34100里面修复。
题目 29
本题目考验对nonlocal
的理解,答案是 D。在前的题目中我们也有所涉及。先看一个例子:
|
这是符合预期的,函数作用域内对 i 的赋值是本地的,不会影响外面的全局变量。之前在题目 26 中提到的 global 方案可以影响外面的全局变量,但是如果是嵌套的作用域呢:
|
使用global
改造:
|
可以看到在最里面使用global
,无论嵌套多少层,都会直接改最外面的那层x = 0
。那么怎么影响到outer
里面的x
呢?这就是nonlocal
的作用:
|
所谓nonlocal
,其实就是说x
不是函数 inner 的本地变量,那么就会向上影响到 outer。也就是影响 Enclosing (嵌套的父级函数的局部) 作用域。关于 Python 的 LEGB 作用域,可以深入搜索了解。
现在思考下,如果nonlocal
的父级没有这个局部变量会继续向父级传播,但是注意,noncal
声明的是函数内的局部变量,父级函数内没有此变量会报错:
|
题目 30
这个题目考查对dataclasses
模块的了解,它的问题在于类 A 的 c 已经定义的默认值,但是继承的 B 却没有。
我们看一下字典的同类问题就能了解:
|
位置参数都要放在关键字参数之前,这个题目稍微有点防水。本来我是想出 Python 3.8 新增的参数类型的约束问题,既然想了就列出来大家感受下:
|
答案是 A。Python 3.8 新增的语法/
约束在/
之前的参数必须在位置上指定并且不能用作关键字参数,*
约束它之后的参数必须使用关键字参数而不能用位置参数。
为了优化 SEO,贴一下报错:
|
延伸阅读
本文来自博客园,作者:码上的生活,转载请注明原文链接:https://www.cnblogs.com/zyl007/p/16795586.html