博客
关于我
一道3行代码的python面试题,我懵逼了...
阅读量:141 次
发布时间:2019-02-27

本文共 1929 字,大约阅读时间需要 6 分钟。

前言

今天来说说交流群里一位群友问的Python题目。刚开始由于没有电脑,我也没有运行出来,之后查阅了一下资料才知道里面的知识还挺多的。废话不多说,我们直接开始。

有意思的题目

题目:写出下面程序运行结果

def multipliers():    return [lambda x:i*x for i in range(4)]print([m(2) for m in multipliers()])

正确答案:

[6,6,6,6]

第一眼看,不就是匿名函数吗?但是仔细想想匿名函数在平时的开发中没怎么用,所以也忘的差不多了。例如那个m(2)当时就不懂是啥意思了。

好,我们就来看看这个题目,首先第一段代码:

def multipliers():    return [lambda x:i*x for i in range(4)]

这是一个典型的列表推导式,简而言之就是在列表中推导计算并且将计算的结果放入列表,上面这串代码我们可以写成:

squares = []for i in range(4):    res = lambda x:i*x    squares.append(res)

这样可能看的更加的简单明了。但是这里面也有一个匿名函数,计算i*x的值,如果大家对匿名函数不太懂的,可以去翻阅相关资料了解一下。

我们接下来看这段代码:

print([m(2) for m in multipliers()])

其实这段代码也很好理解,m(2)的意思就是将2作为参数传入上面的匿名函数当中,但是为什么结果等于[6,6,6,6]呢?

我们来debug一下看看:

断点,开始debug
跳到了multipliers函数当中
仍然在循环体中
运行到3,循环即将结束
跳转下来
重新进入匿名函数计算
计算

这里我省略了几张图,因为结果都是i=3 ,x=2,所以相乘自然为6。

相信大家看到这里仍然有很多的疑问,为什么i最后等于3?

其实这里涉及到闭包函数的概念,什么是闭包函数呢?

当前函数引用到上一层函数局部命名空间的变量时就会触发闭包规则。我们说触发了闭包的函数叫做闭包函数,但是要注意一点:只有当调用闭包函数的时候才会去引用外层函数的变量,因为在调用闭包函数之前,闭包内部的命名空间还不存在。

def multipliers():  squares = []  for i in range(4):      res = lambda x:i*x      squares.append(res)

我们用这种方法来看待这个闭包函数lambda x:i*x,为什么称它为闭包函数呢?因为当执行lambda x:i*x这串代码时调用了上一层函数multipliers()的局部命名空间的变量i,所以此处是闭包函数。

但是此时的列表中并不是匿名函数计算出来的值,因为此时还没有被调用,此时列表中只是有四个匿名函数的内存地址:

[
.
 at 0x1057d1710>, 
.
 at 0x10586dd40>, 
.
 at 0x10586de60>, 
.
 at 0x10586df80>]

那什么时候才算真正调用呢?在上面我们讲过

print([m(2) for m in multipliers()])

这串代码将2作为参数传入上面的匿名函数当中,此时也就是调用了匿名函数。当去调用的时候,for i in range(4)这个循环已经执行完毕,此时的i已经是3了,所以就有了最后的结果:

[6,6,6,6

总结

其实很多题目都是出自一些我们平时不是很重视的知识点上,但是往往这些知识点非常重要。所以学习的时候,对于重点知识不能因为不使用它而去遗忘,而是要多去看看。非常感谢大家能够看完这篇文章,如果在看数超过30,我会写一篇匿名函数专题的文章给大家!

本文部分知识参考了:

https://blog.csdn.net/weixin_34007291/article/details/88755406

近期热门:● 我的神!用Python竟然还能做一个文字套娃● 2020年5月编程语言排行榜,Python竟然排老三● 我珍藏的一些好的Python代码,技巧|上篇● 爬取300本Python书籍,用Python告诉你哪家强?● 卧槽!Pdf转Word用Python轻松搞定!● 我打赌,学会这6招,谁再敢笑你的Python程序慢!

点击阅读原文,阅读菜鸟B站7个原创视频

你可能感兴趣的文章
Mysqldump参数大全(参数来源于mysql5.5.19源码)
查看>>
mysqldump备份时忽略某些表
查看>>
mysqldump实现数据备份及灾难恢复
查看>>
mysqldump数据库备份无法进行操作只能查询 --single-transaction
查看>>
mysqldump的一些用法
查看>>
mysqli
查看>>
MySQLIntegrityConstraintViolationException异常处理
查看>>
mysqlreport分析工具详解
查看>>
MySQLSyntaxErrorException: Unknown error 1146和SQLSyntaxErrorException: Unknown error 1146
查看>>
Mysql_Postgresql中_geometry数据操作_st_astext_GeomFromEWKT函数_在java中转换geometry的16进制数据---PostgreSQL工作笔记007
查看>>
mysql_real_connect 参数注意
查看>>
mysql_secure_installation初始化数据库报Access denied
查看>>
MySQL_西安11月销售昨日未上架的产品_20161212
查看>>
Mysql——深入浅出InnoDB底层原理
查看>>
MySQL“被动”性能优化汇总
查看>>
MySQL、HBase 和 Elasticsearch:特点与区别详解
查看>>
MySQL、Redis高频面试题汇总
查看>>
MYSQL、SQL Server、Oracle数据库排序空值null问题及其解决办法
查看>>
mysql一个字段为空时使用另一个字段排序
查看>>
MySQL一个表A中多个字段关联了表B的ID,如何关联查询?
查看>>