1.名称空间namespace
从名字到对象的一个映射(a mapping from name to objects)
举个例子,在教室里与教学楼的走廊里呼叫Tom,产生的结果是不同的,教室里的Tom回应了,而在走廊里呼叫Tom 多个班级的Tom都回应了,假想的空间,生效的范围
注:为什么random模块的函数需要import,而input /print等函数不需要?
因为python自动引入了常用的名称空间下的函数,称为内建函数
dir(__ builtin__) #查看内建函数
这也是为什么不推荐使用 from a_module import * 的原因,导入的变量可能被当前模块覆盖。
2.scope(作用域):
python 2.0 及之前的版本,python只支持三种作用域(LGB): 局部作用域(Local) ,全局作用域(Global) ,内置作用域(Builtin)
python 2.2 及以上的版本,python加入了一种新的作用域: 嵌套作用域(Enclosing),嵌套作用域可以作为一个选项被开启,本质是对闭包的支持,从而有了四种作用域(LEGB)
1.Local(innermost)
包含局部变量。比如一个函数/方法内部。使用locals()查看当前局部变量
2.Enclosing
包含了非局部(non-local)也非全局(non-global)的变量。
比如两个嵌套函数,内层函数可能搜索外层函数的namespace,但该namespace对内层函数而言既非局部也非全局。
3.Global(next-to-last)
当前脚本的最外层。
比如当前模块的全局变量。
使用globals()查看当前的全局变量
4.Built-in(outtermost)
Python __ builtin__ 模块。
包含了内建的变量/关键字等。
python中能改变变量作用域,引入新的作用域的有:def,class lambda,代码块中的变量在外部不能访问
#1.def类型改变作用域
def test():
tmp=2
print(tmp)
print(tmp) #报错,访问不了,tmp为局部作用域
----------------------------------
#2.Class改变作用域
class test(object):
aa=2017 # 类属性
def __init__(self):
self.aa=33 #实例属性
tst=test()
print(aa) #报错
print(tst.aa) # 33
print(test.aa)
-------------------------------------
#3.lambada 面试题
li = [lambda :x for x in range(10)]
res = li[0]()
print(res)
#输出:9
if/elif/else、try/except/finally、for/while 不引入新的作用域,并不能涉及变量作用域的更改,也就是说他们的代码块中的变量,在外部也是可以访问的
#1.if块中的作用域未改变
if 1 > 0:
name = "hello,Tom"
print(name)
-------------------------------
#2.for 循环作用域未改变
for i in range(10):
age = i
print(age) #打印出"9""
-------------------------------
#3.while循环
while True:
tmp=2017
print(tmp)
break;
print(tmp)
-------------------------------
#4.try--except块,except语句在当前作用域中引入新的变量(异常对象)。
try:
tmp=2017
raise Exception
except:
print(tmp)#可以直接使用哦
注:在c/c#/c++中for-loop中的循环索引是个局部临时变量,通常出了loop块后就销毁,但python中 for-loop 会引入一个新的作用域。for 后面跟着的变量(target list)在循环结束后是不会被删除的,但如果 for 循环的序列为空,这些变量是完全不会被赋值的。
实际上,在Python中,只有模块,类以及函数才会引入新的作用域,其它的代码块是不会引入新的作用域的。
3.LEGB原则
python使用LEGB原则来直接搜索访问对象的namespace
Local -> Enclosing -> Global -> Built-in
意思是:
当有一个变量在 local 域中找不到时,Python会找上一层的作用域,即 enclosing 域(该域不一定存在)。enclosing 域还找不到的时候,再往上一层,搜索模块内的 global 域。最后,会在 built-in 域中搜索。对于最终没有搜索到时,Python会抛出一个 NameError 异常。