【Python】记一次 Python 笔试因“一时没想到xxx”影响的解题错误



笔试是昨天进行的,距现在已经过去了接近 24 h,而且昨晚为了抚平我劳累的大脑,还看了两部电影。因此这里给出的题目和原本可能略不完全一样。

注:本文未完全完成;看出来我在本文介绍的 python 一些语法特性的话,想要了解,请参阅本文给出的“引用”来源。


背景知识

有一题笔试题,我在做的时候脑子里在想什么,这里展示给读者看:

n/a
n/a

上面截图是我大约在3,4 个月前左右看到的1。所以我对于元组这一奇怪的现象印象深刻。

只不过我昨天忘记了上面这个例子在 += 的情况中才会发生。
看下面例子:

>>> l = [2, 3, 4]
>>> l_1 = l + [9, 5]  # l_1 == [2, 3, 4 , 9, 5]
>>> l_2 = l; l_2.extend([9, 5])  # l_2 == [2, 3, 4, 9, 5]

注意上面的例子。
我这里可以问一个问题,但是我还是省略过去,直接说明我要说的重点:
l == l_1 == l_2True
l is l_1False
l is l_2True
也就是说

l = l + [9, 5]

执行完,l 就已经不是原来的 l 了。
所以说,在写代码的时候,可以使用 _list.append()_list.extend() 的时候就不要使用 += 2

  • 在列表中,l += [...]l = l + [...] 不是完全等价的。
    n/a


1 对象,引用和运算

1.1 考你一道笔试

问题:

l = [3, 4]
ll = [l] * 4
print(ll)           # (1)
t = tuple(ll)
print(t)            # (2)
t[0][0] = -8
print(t)            # (3)

以上 (1), (2), (3) 分别输出什么?


经过思考,你可能已经有了你自己的答案。

下面我提供一份我的答案,然后剖析一下它哪里错误,为什么错了,以及一些我们一直在使用,但是却鲜为人知的 Python 秘密。


1.2 我的答案和剖析

这里是我的答案:

(1) [[3, 4], [3, 4], [3, 4], [3, 4]]
(2) ([3, 4], [3, 4], [3, 4], [3, 4])
抛出异常
(3) ([-8, 4], [3, 4], [3, 4], [3, 4])

再看一眼题目,和我的答案对比一下:
n/a

第一个答案(回答)应该来说得出一个“正确的结果”不难。
实际上我还考虑了一下下,因为 l 本来就是数组了,你还给它包一层数组 [l] 是要搞什么鬼,那我最后再 * 一下的结果是要有几层数组(这里数组说的是列表)。
但是我想 * 不要想的太复杂,它其实就是 + 嘛。
所以:
ll = [[3, 4]] + [[3, 4]] + [[3, 4]] + [[3, 4]]

后面你会看到,这种 *+ 的等效切换方式是后面灾难的源头。

然后第二个答案(回答),列表转换为元组,我想这一点几乎没有问题。

最后第三个回答…咦,等等,回答里 抛出错误 是什么鬼?

我想有认真看上面"背景知识"的读者应该猜到我这么给答案的原因是什么了。
在我对“背景知识”里面的元组的“特殊表现”印象深刻之后,我忘了上面说的是在 += 的条件下发生的。
还记得上面我又给出的 l += [83, -9]l.extend([83, -9]) 的例子吗?
在例子中我对 l_1 = l + [83, -9]l_2 = l; l_2.extend(83, -9)is 做了判断。== 比较的实例(对象)的“值”,而 is 比较的是“对象”是否是同一个。

用 C/C++ 语言的说法,对象可以看成指针,而“值”就是指针指向的地址上的值。
所以 == 在 Python 中的运算,用 C/C++ 表示是 *p1 == *p2
is 在 Python 中的运算,用 C/C++ 表示是 p1 == p2
如果你不了解 C/C++,或者对 C/C++ 的指针行为还没有理解透彻,那么按上面比较 “对象的值”相等 和比较 “对象” 是否是同一个 来理解即可。

所以,对于元组来说,“不可变”其实是针对它的“引用”而言的3

写好一份博客有点费心力,目前脑子不太精气神,后续再更新。
注:

  • 回答修改成(把“抛出异常删除”)(3) ([-8, 4], [3, 4], [3, 4], [3, 4]) 还是错误。
    原因:上面说到的 ll = [[3, 4]] + [[3, 4]]l = [3, 4]; ll = [l] * 2 的不同!
    实际应用:构造二维数组(列表)时需要使用深拷贝,对象的默认浅复制行为4
  • t[0] = t[0] + [8, -3]t[0] += [8, -3] 的相同和不同之处(运算符号运行时背后发生的事情)。


2 类的数据属性 - 共享特性

注:
遇到这一笔试题的时候,我个人想到的是编写“单例”的时候使用的“懒汉单例”,而且我使用了 @finally 修饰单例,避开了继承,所以实际编程中,我是特意避开这种编程方式的。

2.1 闷头一锤

给你一题“类的数据属性”以及对其的继承,然后问修改这个变量,结果如何。
我个人在实际编程中特意避开使用“数据属性”,而是对所有需要的类属性在 __init__ 中定义,既然是避开这种编程方式,其实就是没有特定去花时间了解清楚,所以遇到这个问题我真的是吃了一惊。

class Parent(object):
    x = 0

class Child1(Parent):
    pass

class Child2(Parent):
    pass

要点:实例修改 x,类修改 x,子类修改x。


2.2 Answer

共享特性 - 解密一切的一句话:由所有实例共享5

写好一份博客有点费心力,目前脑子不太精气神,后续再更新。
看引用 ‘[^5]’ 书籍中的解释。
注意父类,子类的“命名空间”也是回答正确的/正确思路的一种方式。

个人提醒:共享特性和一直还没去看的 slot 有关系吗?
虽然知道共享特性是这么表现的,不过它的原理什么?是属于“不要问为什么,它就是这么定义的”,还是属于基本语法的衍生表现?


2.3 拓展 - 待续

  • ORM 编程中的应用(实例)。
  • 对于使用需要使用 super().__init__(...) 初始化参数的子类,如果仅仅只是为了调用父类来初始化固定参数,那么可以考虑使用类的实例共享属性这一特性代替调用 super().__init__(...)


3 编译(模块载入)和运行两个阶段

实话说这次笔试出的 Python 题目还真的不简单,都是比较中级偏上的题目。
而且在提问方式上,如果一不小心“忘了xxx”(就是标题说的,一时没想到 xxx),那么必错无疑。

比如上面 ll = [l] * 4 的提问方式换成如何从已有的 l = [2, 3] 创建一个“二维数组”(当然,这种描述方式不够清楚,也比较别扭)来看,更像在问实际应用中会不会犯浅复制的错误。

之所以没有说高级,是因为在我心目中,高级是元编程这一主题,从在 def 函数中返回带有变量属性和方法属性的类实例到类级装饰器再到继承类型检查和创建。这些是我了解过还没有时间和应用场景机会研读的地方。(高级往上可能是 DSL,不过个人目前对此应该无学习和应用的需求)

3.1 似曾相识

问题:

from functools import wraps

def outer(func):
    print("into")
    @wraps(func)
    def inner(**kwargs):
        kwargs.update(a='b')
        result = func(**kwargs)
        return result
    print('outer')
    return inner

@outer
def func(**kwargs):
    return kwargs

print(func(a='I'))
print(func(a='e'))

所以,运行上面代码(可以保存到一个 *.py 文件中,然后运行)得到的输出是什么?


3.2 答案与分析 - 待续

函数级装饰器和闭包之间的关系

写好一份博客有点费心力,目前脑子不太精气神,后续再更新。
要点:

  • 函数装饰器原理:

    @outer
    def func(...):
       ...
    

    等价于

    def func(...):
       ...
    func = outer(func)
    
  • 需要说明清楚一下闭包吗?

  • 本题应用场景:缓存(非 timeit


模块只导入一次 > 1. 共享实例设计 => 模块内全局变量(实例) ;可参考 《python 学习手册》(Learning Python) CH22。



4 字符串行为

4.1 自己加戏的笔试题

今天(2019/Jun/21)有位群友提问了 ?
n/a
这种“结果”是什么原因?

a = "a aa"; b = "a aa"; a is b => False 这种现象确实比较意外,我之前没有见过,而我原本认为相同的字符会是同一块空间(就像 C 语言中的 hard-code 的字符串会分配“常量静态存储区”一样),但是实际上并非如此。

既然之前没有学过,那么刚好在这篇博客内记录一下。


4.2 intern 机制

n/a

Reference:

1. stackoverflow.com: Is there a difference between “==” and “is”? ?
? 这篇讲到了 x = "a"; "aa" is intern(x*2) => True 这一行为。
2. cnblogs.com: Python中字符串的intern机制 ?



总结




Reference


  1. 《流畅的 Python》& 2.6 序列的增量赋值 > 一个关于+=的谜题 ↩︎

  2. 《?》某本 Python “数据结构和算法” 或 “性能优化” 书 ↩︎

  3. 《流畅的 Python》& 8.2.2 元组的相对不可变性 ↩︎

  4. 对象的默认浅复制行为 - 参考(应该是在)《流畅的 Python》一书中(或者是 《Learning Python》) ↩︎

  5. 《Python 学习手册》(Learning Python 4 edition)CH 28 类代码编写细节 > class 语句 > 例子。 ↩︎

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!
提供的源码资源涵盖了小程序应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值