解读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 相关文章推荐
python 输出一个两行字符的变量
Feb 05 Python
python调用cmd命令行制作刷博器
Jan 13 Python
python中的sort方法使用详解
Jul 25 Python
Scrapy-redis爬虫分布式爬取的分析和实现
Feb 07 Python
python实现word 2007文档转换为pdf文件
Mar 15 Python
对python实现模板生成脚本的方法详解
Jan 30 Python
python celery分布式任务队列的使用详解
Jul 08 Python
使用python实现男神女神颜值打分系统(推荐)
Oct 31 Python
python3 字符串知识点学习笔记
Feb 08 Python
python,Java,JavaScript实现indexOf
Sep 09 Python
python操作链表的示例代码
Sep 27 Python
selenium判断元素是否存在的两种方法小结
Dec 07 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之第五天
2006/10/09 PHP
常用PHP封装分页工具类
2017/01/14 PHP
php 将json格式数据转换成数组的方法
2018/08/21 PHP
PHP实现爬虫爬取图片代码实例
2021/03/03 PHP
JQuery打造PHP的AJAX表单提交实例
2009/11/03 Javascript
jquery ajax abort()的使用方法
2010/10/28 Javascript
window.navigate 与 window.location.href 的使用区别介绍
2013/09/21 Javascript
ExpressJS入门实例
2015/01/14 Javascript
JavaScript中的lastIndexOf()方法使用详解
2015/06/06 Javascript
javascript日期比较方法实例分析
2016/06/17 Javascript
AngularJS入门教程之链接与图片模板详解
2016/08/19 Javascript
基于JavaScript 性能优化技巧心得(分享)
2017/12/11 Javascript
基于Vue 2.0的模块化前端 UI 组件库小结
2017/12/21 Javascript
在vue项目中使用Nprogress.js进度条的方法
2018/01/31 Javascript
Vue实现todolist删除功能
2018/06/26 Javascript
vue2.0基于vue-cli+element-ui制作树形treeTable
2019/04/30 Javascript
浅谈Vue的响应式原理
2019/05/30 Javascript
layui的表单提交以及验证和修改弹框的实例
2019/09/09 Javascript
node.js使用mongoose操作数据库实现购物车的增、删、改、查功能示例
2019/12/23 Javascript
vue实现表单未编辑或未保存离开弹窗提示功能
2020/04/08 Javascript
[02:37]2018DOTA2亚洲邀请赛赛前采访 VP.no[o]ne心中最强SOLO是谁
2018/04/04 DOTA
Python实现的简单文件传输服务器和客户端
2015/04/08 Python
Windows下Python的Django框架环境部署及应用编写入门
2016/03/10 Python
python利用dir函数查看类中所有成员函数示例代码
2017/09/08 Python
PyQt5每天必学之组合框
2018/04/20 Python
解决新版Pycharm中Matplotlib图像不在弹出独立的显示窗口问题
2019/01/15 Python
PyQt5内嵌浏览器注入JavaScript脚本实现自动化操作的代码实例
2019/02/13 Python
我用Python抓取了7000 多本电子书案例详解
2019/03/25 Python
Keras Convolution1D与Convolution2D区别说明
2020/05/22 Python
Python使用socket模块实现简单tcp通信
2020/08/18 Python
分布式全文检索引擎ElasticSearch原理及使用实例
2020/11/14 Python
css3实现可滑动跳转的分页插件示例
2014/05/08 HTML / CSS
中科前程Java笔试题
2016/11/20 面试题
跟单文员岗位职责
2014/01/03 职场文书
2014年幼儿园国庆主题活动方案
2014/09/16 职场文书
小学教师自我剖析材料
2014/09/29 职场文书