Python作用域

python作用域

LEGB原则

python中作用域有四种:

L (Local) 局部作用域
E (Enclosing) 闭包函数外的函数中
G (Global) 全局作用域
B (Built-in) 内建作用域

python按照LEGB原则搜索变量,即优先级L>E>G>B。

#dir 为python内建函数
dir = 1 # Global
def outer():
    dir = 2  # Enclosing
    def inner():
        dir = 3 # Local
        return dir
    return inner

print outer()() # 输出3

作用域(Scope)和命名空间(NameSpace)

def/lambda会创建新的作用域,生成器表达式都有引入新的作用域,class的定义没有作用域,只是创建一个隔离的命名空间。在Python中,scope是由namespace按特定的层级结构组合起来的。scope一定是namespace,但namespace不一定是scope。命名空间跟作用域的区别是,它不能在里面再嵌套其他作用域。下面看两个例子。

例1

a = 1
def test():
    a += 1
    a = 2

test() #异常

UnboundLocalError: local variable ‘a’ referenced before assignment。这是因为解释器看到a+=1时,按照LEGB优先在Local中找到了a的声明,执行时先a+=1在a=2声明之前,所有抛出异常。

例2

class A(object):
    x = 2
    gen = (x*i for i in xrange(5))

if __name__ == "__main__":
    a = A()
    print list(a.gen)#异常

上面的代码会抛出异常:NameError: global name ‘x’ is not defined。这是因为gen = (x for _ in xrange(5)是生成器,会产生新的作用域。而classA 中并不产生作用域。按照LEGB原则,不能找到x的定义,所以抛出异常。解决这个问题有几种方案。
1,将x定义为全局变量,这样可以解决异常,但是可能违背了类的逻辑。
2,将生成器表达式改为列表表达式。

gen = [x*i for i in xrange(5)]

在python2中,列表表达式不产生新的作用域,所以不会抛出异常。但是在python3中仍有异常。
3,用A.x的方式访问类属性。

gen = (A.x*i for i in xrange(5))

4,引入lambda函数,将class命名空间的x作为变量传入到匿名函数中。

gen = (lambda x: (x*i for i in xrange(5)))(x)

这个问题可以理解为class不能产生作用域导致的,在函数中就没有这个问题。

def test():
    x = 2
    gen = (i * x for i in xrange(5))
    return gen

gen = test()
print list(gen)#输出[0, 2, 4, 6, 8]
  • 4
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Python中的作用域指的是变量的可访问性和可见性范围。Python按照LEGB原则搜索变量,即优先级L>E>G>B。LEGB分别代表Local(局部作用域)、Enclosing(闭包函数外的函数中)、Global(全局作用域)、Built-in(内置作用域)。在变量引用时,Python会按照LEGB的顺序搜索变量。如果在局部作用域找不到变量,Python会继续搜索封闭函数作用域,再继续搜索全局作用域,最后搜索内置作用域。另外,如果发生变量冲突,可以使用`del`语句从全局作用域中删除变量的定义,从而访问内置作用域中的变量。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [python中的各种作用域](https://blog.csdn.net/weixin_44835655/article/details/127183921)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [Python作用域](https://blog.csdn.net/daiominribao/article/details/53301594)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值