设为首页 收藏本站
查看: 928|回复: 0

[经验分享] Python学习:作用域(namespace)

[复制链接]

尚未签到

发表于 2018-8-14 12:44:16 | 显示全部楼层 |阅读模式
  Python作用域基础
  Python有四个作用域:

  •   L(Local)本地也称作局部作用域;
  •   E(Enclosing)闭包函数外的函数中;
  •   G(global)全局作用域;
  •   B(Built-in)内建作用域;
  变量可以在三个不同的地方分配:

  •   如果一个变量在def内赋值,它被定位在这个函数之内。
  •   如果一个变量在嵌套的def中赋值,对于嵌套的函数来说,它是非本地的。
  •   如果在def之外赋值,它就是整个文件全局的。
  值得注意的是,修改一个对象并不是对一个名称赋值。
  变量名解析:LEGB原则
  对于一个def语句:
  变量名引用分为三个作用域进行查找:首先是本地,之后是函数内(如果有的话),之后全局,最后是内置。L->E->G->B
  Python除了def/class/lambda外,其他如:if/elif/else/  try/except  for/while并不能改变作用域。定义在他们之内的变量,外部还是可以访问。
>>> if True:  
...     a = 'I am A'
  
...
  
>>> a
  
'I am A'
  
# 定义在if语言中的变量a,外部还是可以访问的。
  
# 但是需要注意如果if被 def/class/lambda 包裹,在内部赋值,就变成了此 函数/类/lambda 的局部作用
  在def/class/lambda内进行赋值,就变成了其局部作用域。局部作用域会覆盖全局作用域,但不会影响全局作用域。
g=1            #全局变量  
def func():
  
    g = 2      #局部变量
  
    return g
  

  
print func()    #结果为2
  
print g         #结果为1
  值得注意的是,有时候想再函数内调用全局变量,疏忽了会报错,如下:
#file1  
var = 1
  
def func():
  
    print var
  
    var = 200
  

  
func()
  

  
#file2
  
var = 1
  
def func():
  
    var = var +1
  
    return var
  

  
func()
  

  
#这两个函数都会报错UnboundLocalError: local variable 'var' referenced before assignment
  上述两个函数都会报同样的错误:为赋值之前引用变量!为什么?在函数内部,解释器探测到变量var重新被赋值,所以var变成了局部变量,但是在被赋值之前就使用了var,便会出现这个错误。解决的方法是在函数内部添加globals var语句,但运行函数后全局的var也会被修改。
#file1  
var = 1
  
def func():
  
    global var
  
    print var
  
    var = 200
  

  
func()        #结果为1
  
print var     #全局变量var变为200
  

  
#file2
  
var = 1
  
def func():
  
    global var
  
    var = var +1
  
    return var
  

  
print func()        #结果为2
  闭包Closure
  闭包的定义:如果在一个内部函数里,对外部函数内(不是全局变量)进行引用,那么内部函数就被认为是闭包(closure)。
a = 1  
def external():
  
    global a
  
    a = 200
  
    print a
  
    b =100
  
    def internal():
  
        print b
  
        b = 200
  
        return b
  
    internal()
  
    print b
  

  
print external()
  
#一样会报错,赋值前引用UnboundLocalError: local variable 'b' referenced before assignment
  Python3中有关键字nonlocal可以解决这个问题,但在Python2中尽量不要尝试修改闭包中的变量。
  关于闭包,还有一个坑:
from functools import wraps  

  
def wrapper(log):
  
    def external(F):
  
        @wraps(F)
  
        def internal(**kw):
  
            if False:
  
                log = 'modified'
  
            print log
  
        return internal
  
    return external
  

  
@wrapper('first')
  
def abc():
  
    pass
  

  
print abc()
  也会出现 引用在赋值之前 的错误,原因是解释器探测到了 if False 中的重新赋值,所以不会去闭包的外部函数(Enclosing)中找变量,但 if Flase 不成立没有执行,所以便会出现此错误。除非你还需要else: log='var' 或者 if True 但这样添加逻辑语句就没了意义,所以尽量不要修改闭包中的变量。
  好像用闭包无法实现计数器功能,因为在闭包内部count+=1就会出现在赋值前引用的错误(Python3用关键字nonlocal可以解决)
def counter(start):  
    count = [start]
  
    def internal():
  
        count[0] += 1
  
        return count[0]
  
    return internal
  

  
count = counter(0)
  
for n in range(10):
  
    print count()
  
#结果分别为1,2,3,4,5,6,7,8,9,10
  

  
count = counter(0)
  
print count()
  
#结果为1
  global和globals()
  global用来在函数内部声明全局变量,globals() 和 locals() 提供了基于字典的访问全局和局部变量的方式。
  比如:如果函数1内需要定义一个局部变量,名字另一个函数2相同,但又要在函数1内引用这个函数2。
def var():  
    pass
  

  
def f2():
  
    var = 'Just a String'
  
    f1 = globals()['var']
  
    print var
  
    return type(f1)
  

  
print f2()
  
# Just a String
  
# <type 'function'>

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-551654-1-1.html 上篇帖子: Python基础学习(四) 下篇帖子: Python基本数据分类方式
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表