python命名空间作用域_python 命名空间和作用域

本文详细介绍了Python中的变量作用域与命名空间的关系。命名空间是一个从名字到对象的映射,分为局部、全局和内置三种类型。Python的变量查找顺序遵循LEGB规则:局部->封装->全局->内置。文章通过示例解释了作用域的查找过程,并探讨了类创建的命名空间与作用域的区别。
摘要由CSDN通过智能技术生成

Python中变量作用域的问题和NameSpace(命名空间)有着很大的关系。

命名空间是从名字到对象的一个映射。它其实就是一个dict。具体来说,就是Python为模块,类,对象,函数保存一个__dict__里边就是从名称到对象的映射。Python用命名空间记录变量的轨迹。

在程序运行期间会同时存在多个不同的命名空间,每个命名空间的生命周期也是不同的

命名空间类别:

局部命名空间:每个函数都拥有自己的局部命名空间,记录了函数的局部变量,参数等。

全局命名空间:每个模块都拥有自己的全局命名空间,它记录了一些模块的信息,包括模块级别的变量,常量,类,函数,导入的其他模块等信息。

内置命名空间:任何模块都可以访问它,比如一些内置函数,比如abs(),等其他一些built-in模块。

6e2aca169877abff2835095236d75ba3.png

函数嵌套的情况先的查找顺序:

当前函数的命名空间查找。

父函数的命名空间查找。

模块命名空间查找。

内置命名空间查找。

命名空间的生命周期:

内置命名空间,随着解释器的启动创建,一直存在到解释器运行结束。

全局命名空间,模块被定义读入的时候创建,一般也会存在到解释器运行结束。

局部命名空间,进入一个函数时创建,结束函数调用时候,删除命名空间。

总结一下:

命名空间其实就是一个dict。里边存放着名字到对象的映射。

python当中有不同种类的命名空间,而且查找变量的顺序是局部-->全局-->内置。

python中不同类型的命名空间具有不同的生命周期。

那命名空间和变量作用域有什么关系呢?(Scope一定是NameSpace,但是NameSpace不一定是Scope)

所谓作用域就是指python程序的某一段或者某些段,某个命名空间的名字可以被直接使用,而不是通过(对象.属性)的方式引用。这个作用域就是这个命名空间的作用域。在Python中Scope是按照特定的NameSpace层级结构组织起来的。

Python中存在着4中Scope:

Local:比如一个函数内部的局部变量

Enclosing:比如两个嵌套函数,对于内层函数来说,外层函数命名空间中的变量就是既非全局,又非局部的,即Enclosing的。

Global:比如模块的全局变量。

Built-in:内置对象。

那么Python中是按照什么样的顺序,在作用域中进行变量查找的呢?

Local--->Enclosing--->Global--->Built-in   即著名的LEGB-rule

下面举几个例子:

def test_scope():

inner_var = 10

print inner_var

#会报错NameError: name 'inner_var' is not defined

#这个大家都比较熟悉,和java的一样def关键字会创建一个局部命名空间,它具有local的作用域。

def test_scope():

inner_var = 10

print inner_var

#会报错NameError: name 'inner_var' is not defined

#这个大家都比较熟悉,和java的一样def关键字会创建一个局部命名空间,它具有local的作用域。

def test_scope():

a = True

if True:

b = 1

print b

if __name__ == '__main__':

test_scope()

#这个和java中有不同,在python中,if/elif/else,while,try/except/finally等都不会产生新的命名空间,也不会

产生新的作用域。

def test_scope_outer():

a = 0

b = 1

def test_scope_inner():

print a

print b

return test_scope_inner()

if __name__ == '__main__':

test_scope_outer()

#这个例子也可以正常运行,这个变量查找规则正是复合LEGB-rule。

def test_scope_outer():

a = 0

b = 1

def test_scope_inner():

print a

print b

b = 1

return test_scope_inner()

if __name__ == '__main__':

test_scope_outer()

#UnboundLocalError: local variable 'b' referenced before assignment

#这个例子我只是在内部嵌套方法中后面加了个赋值操作,它就会报上面的错误。这是为什么呢?

因为当执行到内部嵌套方法的print b方法的时候,发现在当前test_scope_inner的local作用域中无法找到变量b,

然后python解释器并没有直接去Enclosing作用域找b,二是尝试着继续往下执行,当碰到b = 1的赋值语句是,它找到了b。

于是就会报变量在赋值前就被引用的错误

class TestScope(object):

a = 3

b = list(a + i for i in range(10))

if __name__ == '__main__':

TestScope()

#NameError: global name 'a' is not defined

#这个又是为什么呢?甚至IDE都不报错误。这个是因为class创建了一个命名空间,但是并没有创建一个Scope。但是下面这个

这个表达式引入了一个新的Local Scope。该Scope不能引用类命名空间定义的变量。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值