作用域规则

命名空间是从名称到对象的映射,Python中主要是通过字典实现的,主要有以下几个命名空间:

内置命名空间,包含一些内置函数和内置异常的名称,在Python解释器启动时创建,一直保存到解释器退出。内置命名实际上存在于一个叫__builtins__的模块中,可以通过globals()['__builtins__'].__dict__查看其中的内置函数和内置异常。

全局命名空间,在读入函数所在的模块时创建,通常情况下,模块命名空间也会一直保存到解释器退出。可以通过内置函数globals()查看。

局部命名空间,在函数调用时创建,其中包含函数参数的名称和函数体内赋值的变量名称。在函数返回或者引发了一个函数内部没有处理的异常时删除,每个递归调用有它们自己的局部命名空间。可以通过内置函数locals()查看。

python解析变量名的时候,首先搜索局部命名空间。如果没有找到匹配的名称,它就会搜索全局命名空间。如果解释器在全局命名空间中也找不到匹配值,最终会检查内置命名空间。如果仍然找不到,就会引发NameError异常。

不同命名空间内的名称绝对没有任何关系,比如:

a = 42

def foo():

a = 13

print "globals: %s" % globals()

print "locals: %s" % locals()

return a

foo()

print "a: %d" % a

结果:

globals: {'a': 42, '__builtins__': , '__file__': 'C:\\Users\\h\\Desktop\\test4.py', '__package__': None, '__name__': '__main__', 'foo': , '__doc__': None}

locals: {'a': 13}

a: 42

可见在函数中对变量a赋值会在局部作用域中创建一个新的局部变量a,外部具有相同命名的那个全局变量a不会改变。

在Python中赋值操作总是在最里层的作用域,赋值不会复制数据,只是将命名绑定到对象。删除也是如此,比如在函数中运行del a,也只是从局部命名空间中删除局部变量a,全局变量a不会发生任何改变。

如果使用局部变量时还没有给它赋值,就会引发UnboundLocalError异常:

a = 42

def foo():

a += 1

return a

foo()

上述函数中定义了一个局部变量a,赋值语句a += 1会尝试在a赋值之前读取它的值,但全局变量a是不会给局部变量a赋值的。

要想在局部命名空间中对全局变量进行操作,可以使用global语句,global语句明确地将变量声明为属于全局命名空间:

a = 42

def foo():

global a

a = 13

print "globals: %s" % globals()

print "locals: %s" % locals()

return a

foo()

print "a: %d" % a

输出:

globals: {'a': 13, '__builtins__': , '__file__': 'C:\\Users\\h\\Desktop\\test4.py', '__package__': None, '__name__': '__main__', 'foo': , '__doc__': None}

locals: {}

a: 13

可见全局变量a发生了改变。

Python支持嵌套函数(闭包),但python 2只支持在最里层的作用域和全局命名空间中给变量重新赋值,内部函数是不可以对外部函数中的局部变量重新赋值的,比如:

def countdown(start):

n = start

def display():

print n

def decrement():

n -= 1

while n > 0:

display()

decrement()

countdown(10)

运行会报UnboundLocalError异常,python 2中,解决这个问题的方法是把变量放到列表或字典中:

def countdown(start):

alist = []

alist.append(start)

def display():

print alist[0]

def decrement():

alist[0] -= 1

while alist[0] > 0:

display()

decrement()

countdown(10)

在python 3中可以使用nonlocal语句解决这个问题,nonlocal语句会搜索当前调用栈中的下一层函数的定义。:

def countdown(start):

n = start

def display():

print n

def decrement():

nonlocal n

n -= 1

while n > 0:

display()

decrement()

countdown(10)

闭包

闭包(closure)是函数式编程的重要的语法结构,Python也支持这一特性,举例一个嵌套函数:

def foo():

x = 12

def bar():

print x

return bar

foo()()

输出:12

可以看到内嵌函数可以访问外部函数定义的作用域中的变量,事实上内嵌函数解析名称时首先检查局部作用域,然后从最内层调用函数的作用域开始,搜索所有调用函数的作用域,它们包含非局部但也非全局的命名。

组成函数的语句和语句的执行环境打包在一起,得到的对象就称为闭包。在嵌套函数中,闭包将捕捉内部函数执行所需要的整个环境。

python函数的code对象,或者说字节码中有两个和闭包有关的对象:

co_cellvars: 是一个元组,包含嵌套的函数所引用的局部变量的名字

co_freevars: 是一个元组,保存使用了的外层作用域中的变量名

再看下上面的嵌套函数:

>>> def foo():

x = 12

def bar():

return x

return bar

>>> foo.func_code.co_cellvars

('x',)

>>> bar = foo()

>>> bar.func_code.co_freevars

('x',)

可以看出外层函数的code对象的co_cellvars保存了内部嵌套函数需要引用的变量的名字,而内层嵌套函数的code对象的co_freevars保存了需要引用外部函数作用域中的变量名字。

在函数编译过程中内部函数会有一个闭包的特殊属性__closure__(func_closure)。__closure__属性是一个由cell对象组成的元组,包含了由多个作用域引用的变量:

>>> bar.func_closure

(,)

若要查看闭包中变量的内容:

>>> bar.func_closure[0].cell_contents

12

如果内部函数中不包含对外部函数变量的引用时,__closure__属性是不存在的:

>>> def foo():

x = 12

def bar():

pass

return bar

>>> bar = foo()

>>> print bar.func_closure

None

当把函数当作对象传递给另外一个函数做参数时,再结合闭包和嵌套函数,然后返回一个函数当做返回结果,就是python装饰器的应用啦。

延迟绑定

需要注意的一点是,python函数的作用域是由代码决定的,也就是静态的,但它们的使用是动态的,是在执行时确定的。

>>> def foo(n):

return n * i

>>> fs = [foo for i in range(4)]

>>> print fs[0](1)

当你期待结果是0的时候,结果却是3。

这是因为只有在函数foo被执行的时候才会搜索变量i的值, 由于循环已结束, i指向最终值3, 所以都会得到相同的结果。

在闭包中也存在相同的问题:

def foo():

fs = []

for i in range(4):

fs.append(lambda x: x*i)

return fs

for f in foo():

print f(1)

返回:

3

3

3

3

解决方法,一个是为函数参数设置默认值:

>>> fs = [lambda x, i=i: x * i for i in range(4)]

>>> for f in fs:

print f(1)

另外就是使用闭包了:

>>> def foo(i):

return lambda x: x * i

>>> fs = [foo(i) for i in range(4)]

>>> for f in fs:

print f(1)

或者:

>>> for f in map(lambda i: lambda x: i*x, range(4)):

print f(1)

使用闭包就很类似于偏函数了,也可以使用偏函数:

>>> fs = [functools.partial(lambda x, i: x * i, i) for i in range(4)]

>>> for f in fs:

print f(1)

这样自由变量i都会优先绑定到闭包函数上。

python函数的规则_Python函数的作用域规则和闭包相关推荐

  1. if python 判断函数返回值_Python函数的返回值和作用域

    函数的返回值和作用域 1.返回值 def guess(x): if x > 3: return "> 3" else: return "<= 3&quo ...

  2. python 倒叙 数组_Python函数合集:68个内置函数请收好!

    内置函数就是python给你提供的, 拿来直接用的函数,比如print.,input等.截止到python版本3.6.2 python一共提供了68个内置函数. #68个内置函数 # abs() di ...

  3. python函数参数列表_python函数列表

    广告关闭 腾讯云11.11云上盛惠 ,精选热门产品助力上云,云服务器首年88元起,买的越多返的越多,最高返5000元! python函数函数是python里组织代码的最小单元,python函数包含以下 ...

  4. python中函数的定义_Python函数是什么_如何定义和调用函数?

    函数是什么? 本节Python培训教程是要大家认识一个"老朋友"--函数. 与数学中的函数不同,在Python中,函数不是看上去冰冷无聊的规则和公式,而是实打实的.有自己作用的代码 ...

  5. python中的作用域以及优先级_Python中的作用域规则详解

    Python是静态作用域语言,尽管它自身是一个动态语言.也就是说,在Python中变量的作用域是由它在源代码中的位置决定的,这与C有些相似,但是Python与C在作用域方面的差异还是非常明显的. 接下 ...

  6. python函数的命名_python函数命名

    广告关闭 腾讯云11.11云上盛惠 ,精选热门产品助力上云,云服务器首年88元起,买的越多返的越多,最高返5000元! 命名空间的生命周期名称空间的生命周期 内置名称空间:(最长)只要 python解 ...

  7. python函数修饰器_Python函数装饰器指南

    Python 具有强大的功能和富有表现力的语法.我最喜欢的装饰之一.在设计模式的上下文中,装饰器动态更改方法或类功能,而不必直接使用子类.当您需要扩展功能,但不想修改原函数时,这是理想的选择.我们可以 ...

  8. python average函数详解_python 函数详解

    函数函数是代码的一种组织形式 函数应该能完成一项特定的工作,而且一般一个函数只完成一项工作 有些语言,分函数和过程两个概念,通俗解释是,有返回结果的是函数,无返回结果的叫过程,python不加以区分 ...

  9. python函数进阶小结_python函数的进阶

    形参角度: 万能参数 如果我们在传参数时不很清楚有哪些,或者说给一个函数传了很多实参,考虑用动态参数也叫万能参数. 万能参数,即动态参数,分为两种:动态接收位置参数 *args,动态接收关键字参数** ...

  10. python修饰符作用_python函数修饰符@的使用

    python函数修饰符@的作用是为现有函数增加额外的功能,常用于插入日志.性能测试.事务处理等等. 创建函数修饰符的规则: (1)修饰符是一个函数 (2)修饰符取被修饰函数为参数 (3)修饰符返回一个 ...

最新文章

  1. [WIKIOI1298]凸包周长[裸凸包]
  2. 友商惭愧不?2000+的手机红米直接干到千元
  3. 语言速算24点的小窍门_生活百科,知道这25个实用的居家小窍门,可以为你省去不少麻烦...
  4. 【期末划重点】电路与电子技术基础
  5. C++ 智能指针和工厂模式
  6. python-第三方接口获取验证码
  7. web网页设计实例作业 我的家乡- 达州(4页) HTML+CSS+JavaScript dreamweaver作业静态HTML网页设计模板
  8. python对excel中需要的数据的单元格填充颜色
  9. 数据库系统的基本组成
  10. GC算法精解(五分钟让你彻底明白标记/清除算法)
  11. matlab移植linux qt总结 (未实现)(21.8.25 已另寻别径)
  12. LeetCode 全站第一,牛逼!
  13. APP设计阅读:6本交互设计殿堂级的书籍
  14. 将你的现实生活照片变成卡通头像
  15. 【NLP】NLP中的对抗训练
  16. linux shell 中数组的定义和for循环遍历
  17. ssm框架搭建之ss框架
  18. linux中交换分区,linux中的交换分区(swap)及优化
  19. Android KK平台的一个bug----在收到内容只有一个“=”的信息后,手机自动重启
  20. 参观场景下语音播报的特殊处理

热门文章

  1. 【OpenGL】Shader实例分析(七)- 雪花飘落效果
  2. 语音分享应用ios源码项目
  3. Cloudera Manager和CDH4.1的安装
  4. Stopwatch示例
  5. 深入浅出JavaScript (四)DHTML
  6. Eclipse安装插件支持jQuery智能提示
  7. 打造专业化FTTD布线产品解决方案
  8. hisi mmz模块驱动讲解
  9. linux mysql 管理员,Linux管理员-CentOS 7上的MySQL设置
  10. Linux电源管理-概述