解读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 相关文章推荐
本地文件上传到七牛云服务器示例(七牛云存储)
Jan 11 Python
python动态性强类型用法实例
May 09 Python
Java中重定向输出流实现用文件记录程序日志
Jun 12 Python
浅析Python 3 字符串中的 STR 和 Bytes 有什么区别
Oct 14 Python
python 实现调用子文件下的模块方法
Dec 07 Python
python从子线程中获得返回值的方法
Jan 30 Python
Python高级特性 切片 迭代解析
Aug 23 Python
Python 处理文件的几种方式
Aug 23 Python
python OpenCV GrabCut使用实例解析
Nov 11 Python
python元组的概念知识点
Nov 19 Python
Python数组拼接np.concatenate实现过程
Apr 18 Python
解决Python 函数声明先后顺序出现的问题
Sep 02 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加水印的代码(支持半透明透明打水印,支持png透明背景)
2013/01/17 PHP
phplist及phpmailer(组合使用)通过gmail发送邮件的配置方法
2016/03/30 PHP
Laravel5中防止XSS跨站攻击的方法
2016/10/10 PHP
对xmlHttp对象的理解
2011/01/17 Javascript
javascript检测是否联网的实现代码
2014/09/28 Javascript
js实现缓冲运动效果的方法
2015/04/10 Javascript
JavaScript学习笔记之取数组中最大值和最小值
2016/03/23 Javascript
Javascript获取图片原始宽度和高度的方法详解
2016/09/20 Javascript
Vue生命周期示例详解
2017/04/12 Javascript
JS利用cookies设置每隔24小时弹出框
2017/04/20 Javascript
js 倒计时(高效率服务器时间同步)
2017/09/12 Javascript
vuex与组件联合使用的方法
2018/05/10 Javascript
纯JS实现的读取excel文件内容功能示例【支持所有浏览器】
2018/06/23 Javascript
KOA+egg.js集成kafka消息队列的示例
2018/11/09 Javascript
微信小程序rich-text富文本用法实例分析
2019/05/20 Javascript
Vue.js实现备忘录功能
2019/06/26 Javascript
Vue 使用beforeEach实现登录状态检查功能
2019/10/31 Javascript
JS插件amCharts实现绘制柱形图默认显示数值功能示例
2019/11/26 Javascript
d3.js实现图形拖拽
2019/12/19 Javascript
[02:35]DOTA2超级联赛专访XB 难忘一年九冠称王
2013/06/20 DOTA
测试、预发布后用python检测网页是否有日常链接
2014/06/03 Python
使用 python pyautogui实现鼠标键盘控制功能
2019/08/04 Python
关于Python-faker的函数效果一览
2019/11/28 Python
Python连接字符串过程详解
2020/01/06 Python
Python3 io文本及原始流I/O工具用法详解
2020/03/23 Python
伊利莎白雅顿官网:Elizabeth Arden
2016/10/10 全球购物
SISLEY希思黎官方旗舰店:享誉全球的奢华植物美容品牌
2018/04/25 全球购物
德国家具折扣店:POCO
2020/02/28 全球购物
存储过程和sql语句的优缺点
2014/07/02 面试题
mysql有关权限的表都有哪几个
2015/04/22 面试题
2014年文明创建工作总结
2014/11/25 职场文书
2016年乡镇七一建党节活动总结
2016/04/05 职场文书
OpenCV-Python直方图均衡化实现图像去雾
2021/06/07 Python
浅谈Python协程asyncio
2021/06/20 Python
SQL注入详解及防范方法
2021/12/06 MySQL
Oracle 11g数据库使用expdp每周进行数据备份并上传到备份服务器
2022/06/28 Oracle