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

[经验分享] python中变量作用域及嵌套作用域

[复制链接]

尚未签到

发表于 2017-5-5 09:34:26 | 显示全部楼层 |阅读模式
在python中,变量查找遵循LGB原则,即优先在局部作用域(local scope)中对变量进行查找,失败则在全局作用域(global scope)中进行查找,最后尝试再内建作用域(build-in scope)内查找,如果还是未找到的话,则抛出异常。后来由于闭包和嵌套函数的出现,作用域又增加了外部作用域,这样变量的查找作用域优先级变为:局部、外部、全局和内建。 作用域由def、class、lambda等语句产生,if、try、for等语句并不会产生新的作用域。变量名引用分为三个作用域进行查找:首先是本地,然后是函数内(如果有的话),之后是全局,最后是内置。在默认情况下,变量名赋值会创建或者改变本地变量。全局声明将会给映射到模块文件内部的作用域的变量名赋值。Python
的变量名解析机制也称为 LEGB 法则,具体如下:当在函数中使用未确定的变量名时,Python 搜索 4 个作用域:本地作用域(L),之后是上一层嵌套结构中 def 或 lambda 的本地作用域(E),之后是全局作用域(G),最后是内置作用域(B)。按这个查找原则,在第一处找到的地方停止。如果没有找到,Python 会报错的。

a = 1
def f():
a = 2
def g():
print a //[1]:输出结果为2
return g
func = f()
func()//[2]




代码的[2]处调用的函数实际上调用的是函数f 中定义的内嵌函数g,在代码 的[1]处,函数g 内的“print a”的输出结果为2。初看上去有些疑问,因为函数f 内的约束“a = 2”在其之外应该是不起作用的,当执行func()时,起作用的约束应该是“a = 1”才对。但是我们之前说到了,作用域仅仅是由文本决定的,函数g 位于函数f 之内,所以函数g 定义的作用域内嵌于函数f 的作用域之内。换句话说,函数f 的作用域是函数g 的作用域的直接外围作用域,所以,按照最内嵌套作用域规则,[1]处的名字引用应该引用的是函数f
定义的作用域中所创建的约束。


尽管在代码清单8-2 的[2]处,“a = 2”这个约束已经不起作用了,但是Python 在执行“func = f()”时,会执行函数f 中的“def g():”语句,这时Python 会将约束“a = 2”与函数g 对应的函数对象捆绑在一起,将捆绑后的结果返回,这个捆绑起来的整体被称为“闭包”。


实际上这里有一个相当微妙的问题,最内嵌套作用域规则是“闭包”的结果呢,还是“闭包”是最内嵌套作用域规则的实现方案?这两个问题看上去是一致的,但却隐含着谁决定谁的关系。实际上,Python 实现闭包是为了实现最内嵌套作用域规则。换句话说,最内嵌套作用域规则是语言设计时的设计策略,即是形而上的“道”;而闭包则实现语言时的一种方案,即是形而下的“器”。

python能够改变变量作用域的代码段是def、class、lamda.
def scopetest():  
localvar=6;  
print(localvar)  
scopetest()  
#print(localvar) #去除注释这里会报错,因为localvar是本地变量  
if/elif/else、try/except/finally、for/while
while True:  
newvar=8  
print(newvar)  
break;  
print(newvar)  
try:  
newlocal=7  
raise Exception  
except:  
print(newlocal)#可以直接使用哦  
输出结果:8 8 7
可见这个关键字中定义变量,他们的作用域跟外部是一致的,这个跟Java的作用域概念有点不一样。
变量搜索路径是:本地变量->全局变量
def scopetest():  
var=6;  
print(var)#  
var=5   
print(var)  
scopetest()  
print(var)  
输出结果:5 6 5
这里var 首先搜索的是本地变量,scopetest()中 var=6相当于自己定义了一个局部变量,赋值为6. 当然如果的确要修改全局变量的值,则需要如下:
def scopetest():  
global var   
var=6;  
print(var)#  
var=5   
print(var)  
scopetest()  
print(var)  
输出结果:5 6 6
再看一种这种情况:
def scopetest():  
var=6;  
print(var)#  
def innerFunc():  
print(var)#look here  
innerFunc()  
var=5   
print(var)  
scopetest()  
print(var)  
输出结果:5 6 6 5

运维网声明 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-373268-1-1.html 上篇帖子: Python 下篇帖子: [Python]Invalid mode ('w') or filename: 'd:\x0bideo'
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

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

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

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

扫描微信二维码查看详情

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


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


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


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



合作伙伴: 青云cloud

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