Python中的作用域规则详解


Posted in Python onJanuary 30, 2015

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

接下来会谈论Python的作用域规则,在这中间也会说明一下Python与C在作用域方面的不同。

在Python 2.0及之前的版本中,Python只支持3种作用域,即局部作用域,全局作用域,内置作用域;在Python 2.2中,Python正式引入了一种新的作用域 --- 嵌套作用域;在Python 2.1中,嵌套作用域可以作为一个选项被开启;嵌套作用域的引入,本质上为Python实现了对闭包的支持,关于闭包的知识,网上有很多解释,这里就不详细展开了。相应地,变量查找顺序由之前的LGB变成LEGB(L:Local,E:Enclosing,G:Global,B:Built-in)。

在Python中,并不是任何代码块都能引入新的作用域,这与C有很大的不同:

#include<stdio.h>

int main() {

    if(2 > 0) {

        int i = 0;

    }

    printf("i = %d", i);

    return 0;

}

在这段代码中,if子句引入了一个局部作用域,变量i就存在于这个局部作用域中,但对外不可见,因此,接下来在printf函数中对变量i的引用会引发编译错误。

但是,在Python中却并非如此:

if True:

    i = 0

print i

 在这段代码中,if子句并没有引入一个局部作用域,变量i仍然处在全局作用域中,因此,变量i对于接下来的print语句是可见的。

实际上,在Python中,只有模块,类以及函数才会引入新的作用域,其它的代码块是不会引入新的作用域的。

在Python中,使用一个变量之前不必预先声明它,但是在真正使用它之前,它必须已经绑定到某个对象;而名字绑定将在当前作用域中引入新的变量,同时屏蔽外层作用域中的同名变量,不论这个名字绑定发生在当前作用域中的哪个位置。

def f():

    print i

f()

运行结果将显示:NameError: global name 'i' is not defined。Python首先在函数f的本地作用域中查找变量i,查找失败,接着在全局作用域和内置作用域中查找变量i,仍然失败,最终抛出NameError异常。

i = 0

def f():

    i = 8

    print i

f()

print i

运行结果显示:8和0。i = 8是一个名字绑定操作,它在函数f的局部作用域中引入了新的变量i,屏蔽了全局变量i,因此f内部的print语句看到的是局部变量i,f外部的print语句看到的是全局变量i。

i = 0

def f():

    print i

    i = 0

f()

运行结果显示:UnboundLocalError: local variable 'i' referenced before assignment。在这个例子当中,函数f中的变量i是局部变量,但是在print语句使用它的时候,它还未被绑定到任何对象之上,所以抛出异常。

print i

i = 0

不论是以交互的方式运行,还是以脚本文件的方式运行,结果都显示:NameError: name 'i' is not defined。这里的输出结果又与上一个例子不同,这是因为它在顶级作用域(模块作用域)的缘故。对于模块代码而言,代码在执行之前,没有经过什么预处理,但是对于函数体而言,代码在运行之前已经经过了一个预处理,因此不论名字绑定发生在作用域的那个位置,它都能感知出来。Python虽然是一个静态作用域语言,但是名字查找确实动态发生的,因此直到运行的时候,才会发现名字方面的问题。

在Python中,名字绑定在所属作用域中引入新的变量,同时绑定到一个对象。名字绑定发生在以下几种情况之下:

1.参数声明:参数声明在函数的局部作用域中引入新的变量;
2.赋值操作:对一个变量进行初次赋值会在当前作用域中引入新的变量,后续赋值操作则会重新绑定该变量;
3.类和函数定义:类和函数定义将类名和函数名作为变量引入当前作用域,类体和函数体将形成另外一个作用域;
4.import语句:import语句在当前作用域中引入新的变量,一般是在全局作用域;
5.for语句:for语句在当前作用域中引入新的变量(循环变量);
6.except语句:except语句在当前作用域中引入新的变量(异常对象)。

在Python中,类定义所引入的作用域对于成员函数是不可见的,这与C++或者Java是很不同的,因此在Python中,成员函数想要引用类体定义的变量,必须通过self或者类名来引用它。

嵌套作用域的加入,会导致一些代码编译不过或者得到不同的运行结果,在这里Python解释器会帮助你识别这些可能引起问题的地方,给出警告。

locals函数返回所有的局部变量,但是不会返回嵌套作用域中的变量,实际上没有函数会返回嵌套作用域中的变量。

Python 相关文章推荐
推荐下python/ironpython:从入门到精通
Oct 02 Python
python使用urllib2模块获取gravatar头像实例
Dec 18 Python
在Python中使用poplib模块收取邮件的教程
Apr 29 Python
python中实现精确的浮点数运算详解
Nov 02 Python
python中使用iterrows()对dataframe进行遍历的实例
Jun 09 Python
对Python 窗体(tkinter)文本编辑器(Text)详解
Oct 11 Python
python添加模块搜索路径和包的导入方法
Jan 19 Python
python中的global关键字的使用方法
Aug 20 Python
python破解bilibili滑动验证码登录功能
Sep 11 Python
Python使用Pygame绘制时钟
Nov 29 Python
python爬虫破解字体加密案例详解
Mar 02 Python
Python可变与不可变数据和深拷贝与浅拷贝
Apr 06 Python
Python中使用Boolean操作符做真值测试实例
Jan 30 #Python
Python中的zip函数使用示例
Jan 29 #Python
Python的另外几种语言实现
Jan 29 #Python
python中使用xlrd、xlwt操作excel表格详解
Jan 29 #Python
Python中实现常量(Const)功能
Jan 28 #Python
Python使用random和tertools模块解一些经典概率问题
Jan 28 #Python
Python中的异常处理学习笔记
Jan 28 #Python
You might like
PHP中读写文件实现代码
2011/10/20 PHP
PHP获取文件相对路径的方法
2015/02/26 PHP
PHP单例模式详细介绍
2015/07/01 PHP
Yii框架响应组件用法实例分析
2019/09/04 PHP
破解Session cookie的方法
2006/07/28 Javascript
js 动态选中下拉框
2009/11/26 Javascript
jquery 快速回到页首的方法
2013/12/05 Javascript
jquery鼠标停止移动事件
2013/12/21 Javascript
document.forms[].submit()使用介绍
2014/02/19 Javascript
弹出窗口并且此窗口带有半透明的遮罩层效果
2014/03/13 Javascript
删除条目时弹出的确认对话框
2014/06/05 Javascript
JavaScript闭包详解
2015/02/02 Javascript
JS+JSP通过img标签调用实现静态页面访问次数统计的方法
2015/12/14 Javascript
jQuery实现本地预览上传图片功能
2016/01/08 Javascript
如何处理JSON中的特殊字符
2016/11/30 Javascript
详解Vue.js动态绑定class
2016/12/20 Javascript
jquery 仿锚点跳转到页面指定位置的实例
2017/02/14 Javascript
Node.js获取前端ajax提交的request信息
2017/02/20 Javascript
vue.js 获取当前自定义属性值
2017/06/01 Javascript
Vue如何实现组件的源码解析
2017/06/08 Javascript
vue-cli3.0如何使用CDN区分开发、生产、预发布环境
2018/11/22 Javascript
用 Python 连接 MySQL 的几种方式详解
2018/04/04 Python
matplotlib 纵坐标轴显示数据值的实例
2018/05/25 Python
Python实现定制自动化业务流量报表周报功能【XlsxWriter模块】
2019/03/11 Python
Python TCPServer 多线程多客户端通信的实现
2019/12/31 Python
Python字典深浅拷贝与循环方式方法详解
2020/02/09 Python
Pandas实现一列数据分隔为两列
2020/05/18 Python
你所知道的集合类都有哪些?主要方法?
2012/12/31 面试题
心理健康课教学反思
2014/02/13 职场文书
电气工程自动化求职信
2014/03/14 职场文书
小学生母亲节演讲稿
2014/05/07 职场文书
2014年感恩母亲演讲稿
2014/05/27 职场文书
2014年培训工作总结范文
2014/11/27 职场文书
redis复制有可能碰到的问题汇总
2022/04/03 Redis
css3 文字断裂效果
2022/04/22 HTML / CSS
MySQL新手入门进阶语句汇总
2022/09/23 MySQL