解读Python编程中的命名空间与作用域


Posted in Python onOctober 16, 2015

变量是拥有匹配对象的名字(标识符)。命名空间是一个包含了变量名称们(键)和它们各自相应的对象们(值)的字典。
一个Python表达式可以访问局部命名空间和全局命名空间里的变量。如果一个局部变量和一个全局变量重名,则局部变量会覆盖全局变量。
每个函数都有自己的命名空间。类的方法的作用域规则和通常函数的一样。
Python会智能地猜测一个变量是局部的还是全局的,它假设任何在函数内赋值的变量都是局部的。
因此,如果要给全局变量在一个函数里赋值,必须使用global语句。
global VarName的表达式会告诉Python, VarName是一个全局变量,这样Python就不会在局部命名空间里寻找这个变量了。

命名空间的定义
Python命名空间是名称到对象的映射,这就像是字典,键名是变量名,值是变量的值。比如:

>>> x = 3
>>> globals()
{'__builtins__': <module '__builtin__' (built-in)>, '__name__': '__main__', '__doc__': None, 'x': 3, '__package__': None}

可以看到变量x,3以字典的形式存放在globals空间内。以之对应的名字空间还有:locals()。

>>> locals()
{'__builtins__': <module '__builtin__' (built-in)>, '__name__': '__main__', '__doc__': None, 'x': 3, '__package__': None}

实际上,你可以通过向名字添加键名和值:

>>> globals()['y'] = 5
>>> y
5

上图左侧是内置命名空间,右侧是不同的模块,有各自的全局命名空间,全局命名空间内定义函数就会有局部命名空间。

命名空间的种类
Python中有三种命名空间:

a) 局部,函数内的命名空间就是局部的;
b) 全局,模块内的命名空间就是全局的;

c) 内置,包括异常类型、内建函数和特殊方法,可以代码中任意地方调用;

下面讨论关于名字空间的搜索顺序,先来看张图:

解读Python编程中的命名空间与作用域

命名空间的可见性(作用域)
a) 内置命名空间在代码所有位置都是可见的,所以可以随时被调用;

b) 全局命名空间和局部命名空间中, 如果有同名变量,在全局命名空间处,局部命名空间内的同名变量是不可见的;

c) 在局部命名空间处,全局命名空间的同名变量是不可见的(只有变量不同名的情况下,可使用 global关键字让其可见)。

知道了可见性,下面说变量的查找顺序就要清楚多了。

命名空间的查找顺序
a) 如果在函数内调用一个变量,先在函数内(局部命名空间)查找,如果找到则停止查找。否则在函数外部(全局命名空间)查找,如果还是没找到,则查找内置命名空间。如果以上三个命名都未找到,则抛出NameError 的异常错误。
b) 如果在函数外调用一个变量,则在函数外查找(全局命名空间,局部命名空间此时不可见),如果找到则停止查找,否则到内置命名空间中查找。如果两者都找不到,则抛出异常。只有当局部命名空间内,使用global 关键字声明了一个变量时,查找顺序则是 a) 的查找顺序。

为了帮助理解,来举个例子,我们在全局命名空间里定义一个变量money。我们再在函数内给变量money赋值,然后Python会假定money是一个局部变量。然而,我们并没有在访问前声明一个局部变量money,结果就是会出现一个UnboundLocalError的错误。取消global语句的注释就能解决这个问题。

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
Money = 2000
def AddMoney():
  # 想改正代码就取消以下注释:
  # global Money
  Money = Money + 1
 
print Money
AddMoney()
print Money

dir()函数
dir()函数一个排好序的字符串列表,内容是一个模块里定义过的名字。
返回的列表容纳了在一个模块里定义的所有模块,变量和函数。如下一个简单的实例:

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
# 导入内置math模块
import math
 
content = dir(math)
 
print content;

以上实例输出结果:

['__doc__', '__file__', '__name__', 'acos', 'asin', 'atan', 
'atan2', 'ceil', 'cos', 'cosh', 'degrees', 'e', 'exp', 
'fabs', 'floor', 'fmod', 'frexp', 'hypot', 'ldexp', 'log',
'log10', 'modf', 'pi', 'pow', 'radians', 'sin', 'sinh', 
'sqrt', 'tan', 'tanh']

在这里,特殊字符串变量__name__指向模块的名字,__file__指向该模块的导入文件名。

globals()和locals()函数

  • 根据调用地方的不同,globals()和locals()函数可被用来返回全局和局部命名空间里的名字。
  • 如果在函数内部调用locals(),返回的是所有能在该函数里访问的命名。
  • 如果在函数内部调用globals(),返回的是所有在该函数里能访问的全局名字。
  • 两个函数的返回类型都是字典。所以名字们能用keys()函数摘取。

reload()函数
当一个模块被导入到一个脚本,模块顶层部分的代码只会被执行一次。
因此,如果你想重新执行模块里顶层部分的代码,可以用reload()函数。该函数会重新导入之前导入过的模块。语法如下:

reload(module_name)

在这里,module_name要直接放模块的名字,而不是一个字符串形式。比如想重载hello模块,如下:

reload(hello)
Python 相关文章推荐
用C++封装MySQL的API的教程
May 06 Python
Python学习pygal绘制线图代码分享
Dec 09 Python
python之django母板页面的使用
Jul 03 Python
python实现手机销售管理系统
Mar 19 Python
使用python获取(宜宾市地震信息)地震信息
Jun 20 Python
python 数据提取及拆分的实现代码
Aug 26 Python
python中对_init_的理解及实例解析
Oct 11 Python
python小项目之五子棋游戏
Dec 26 Python
Python matplotlib实时画图案例
Apr 23 Python
python集合的新增元素方法整理
Dec 07 Python
Python使用openpyxl复制整张sheet
Mar 24 Python
详解Python中__new__方法的作用
Mar 31 Python
Python中的模块导入和读取键盘输入的方法
Oct 16 #Python
Python中基本的日期时间处理的学习教程
Oct 16 #Python
简单介绍使用Python解析并修改XML文档的方法
Oct 15 #Python
Python中将字典转换为XML以及相关的命名空间解析
Oct 15 #Python
详细解读Python中解析XML数据的方法
Oct 15 #Python
深入解析Python编程中JSON模块的使用
Oct 15 #Python
使用Python解析JSON数据的基本方法
Oct 15 #Python
You might like
php源码分析之DZX1.5字符串截断函数cutstr用法
2015/06/17 PHP
php实现多城市切换特效
2015/08/09 PHP
Yii2如何批量添加数据
2016/05/17 PHP
PHP示例演示发送邮件给某个邮箱
2019/04/03 PHP
关于laravel模板中生成URL的几种模式总结
2019/10/18 PHP
PHP中abstract(抽象)、final(最终)和static(静态)原理与用法详解
2020/06/05 PHP
javascript入门·图片对象(无刷新变换图片)\滚动图像
2007/10/01 Javascript
在javascript中对于DOM的加强
2013/04/11 Javascript
深入理解javaScript中的事件驱动
2013/05/21 Javascript
js调用AJAX时Get和post的乱码解决方法
2013/06/04 Javascript
javascript拖拽上传类库DropzoneJS使用方法
2013/12/05 Javascript
js中继承的几种用法总结(apply,call,prototype)
2013/12/26 Javascript
js中switch case循环实例代码
2013/12/30 Javascript
用JS实现图片轮播效果代码(一)
2016/06/26 Javascript
javascript设计模式Constructor(构造器)模式
2016/08/19 Javascript
详解js的事件处理函数和动态创建html标记方法
2016/12/16 Javascript
prototype与__proto__区别详细介绍
2017/01/09 Javascript
百度地图JavascriptApi Marker平滑移动及车头指向行径方向
2017/03/13 Javascript
解决js ajax同步请求造成浏览器假死的问题
2018/01/18 Javascript
jquery css实现流程进度条
2020/03/26 jQuery
如何在vue-cli中使用css-loader实现css module
2021/01/07 Vue.js
[01:07:19]2018DOTA2亚洲邀请赛 4.5 淘汰赛 Mineski vs VG 第一场
2018/04/06 DOTA
python生成指定长度的随机数密码
2014/01/23 Python
Python实现模拟时钟代码推荐
2015/11/08 Python
python re模块的高级用法详解
2018/06/06 Python
Python3导入CSV文件的实例(跟Python2有些许的不同)
2018/06/22 Python
python统计指定目录内文件的代码行数
2019/09/19 Python
Python通过4种方式实现进程数据通信
2020/03/12 Python
python 输入字符串生成所有有效的IP地址(LeetCode 93号题)
2020/10/15 Python
德国W家官网,可直邮中国的母婴商城:Windeln.de
2021/03/03 全球购物
北京麒麟网信息技术有限公司网络游戏测试面试题
2013/09/28 面试题
新春寄语大全
2014/04/09 职场文书
医院科室评语
2015/01/04 职场文书
英语感谢信范文
2015/01/20 职场文书
为什么在foreach循环中JAVA集合不能添加或删除元素
2021/06/11 Java/Android
vue使用watch监听属性变化
2022/04/30 Vue.js