Python坑:bool是int的子类、列表循环中的变量泄露、lambda在闭包中会保存局部变量、重用全局变量

bool是int的子类

a = True
print isinstance(a, int)
print True == 1
print False == 0

运行结果:

True
True
True

列表循环中的变量泄露

# 情况一    
i = 1
li = [i for i in range(5)]
print i
# 情况二
i = 1
for i in range(5):
    pass
print i

运行结果:

4
4

列表推导式和for循环对于它们的迭代变量没有私有的作用域,为了避免泄漏变量的问题,不要在循环中使用外部同名变量。

lambda在闭包中会保存局部变量

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:778463939
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
alist = [lambda : i for i in range(5)]
print alist
for j in alist:
    print j()

运行结果:

4
4
4
4
4

原因是当赋值给alist的时候,lambda表达式就执行了i循环,直到 i =4,i会保留。

问题的本质在与python中的属性查找规则,LEGB(local,enclousing,global,bulitin),在上面的例子中,i就是在闭包作用域(enclousing),而Python的闭包是迟绑定,这意味着闭包中用到的变量的值,是在内部函数被调用时查询得到的。

# 解决方法1:换成生成器
alist = (lambda : i for i in range(5))
for j in alist:
    print j()

# 解决方法2:变闭包作用域为局部作用域
alist = [lambda i=i: i for i in range(5)]
for j in alist:
    print j()

运行结果:

0
1
2
3
4

重用全局变量

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:778463939
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
x=10
def demo():
    print x
    x=20
    print x
demo()

运行结果:

UnboundLocalError: local variable 'x' referenced before assignment

原因是某个作用域只要有该变量的赋值语句,该变量就是局部变量。在赋值之前局部变量不存在,所以报错。

解决方法:

x=10
def demo():
    global x  # 使用前加全局变量,后果是x变成了全局变量
    print x
    x=20
    print x
demo()

运行结果:

10
20
上一篇:93. 复原 IP 地址


下一篇:Python的内存管理机制