当前位置:首页 > 编程问答 >

Python的作用域是干嘛的

提问时间:2020年06月17日 10:55:38问题被 143 网友关注
共 1 个关于“Python的作用域怎么理解”精选答案
邱锦程
推荐于:2020-06-17
最佳回答

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
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函数返回所有的局部变量,但是不会返回嵌套作用域中的变量,实际上没有函数会返回嵌套作用域中的变量。

参考资料

  • Python3程序开发指南出版社:人民邮电出版社 作者:萨默菲尔德
  • python算法教程出版社:人民邮电出版社 作者:赫特兰
  • Python即学即用出版社:机械工业出版社 作者:张燕妮
  • Python黑客攻防入门出版社:人民邮电出版社 作者:赵诚文
  • 人人都懂设计模式:从生活中领悟设计模式(Python实现)出版社:电子工业出版社 作者:罗伟富
  • 相关问题

  • python os.chown() 方法及作用02-21
  • Python中的逗号作用是什么06-17
  • 实例分析python fileisatty()方法的定义及作用02-02
  • python使用协程与并发的作用以及优缺点09-25
  • python final作用域的代码一定会被执行吗05-18
  • Python运算符的作用与意义03-24
  • Python中模块导入时全局变量__all__的作用02-12
  • 介绍Python中的__init__的作用01-24
  • 单例模式python的作用整理01-03
  • 更多答案
    网友NO.259278

    05-18

    python final作用域的代码一定会被执行吗, 正常的情况下,finally作用域的代码一定会被执行的,不管是否发生异常。哪怕是调用了sys.exit函数,finally也是会被执行的。 那怎么样才能让finally代码不执行了。 import timechoice = Truetry: if choice: while True: pass else: print Please pull the plug on your computer sometime soon... time.sleep(60 * 60 * 24 * 365)finally: print Finally ... 上面的代码主要是通过让流程停滞在try作用域里,从而实现了需求。 内容扩展 变量的作用域 在Python程序中创建、改变、查找变量名时,都是在一个保存变量名的空间中进行,我们称之为命名空间,也被称之为作用域。python的作用域是静态的,在源代码中变量名被赋值的位置决定了该变量能被访问的范围。即……

    网友NO.646742

    06-26

    python变量的作用域, 变量作用域: 一般在函数体外定义的变量成为全局变量,在函数内部定义的变量称为局部变量。 全局变量所有作用域都可读,局部变量只能在本函数可读 函数在读取变量时,优先读取函数本身自有的局部变量,再去读全局变量 全局变量 读,均可读 赋值,global 字典,列表可修改 全局变量全大写 例如 name = 'Tim' #全局变量def f1(): age = 18 #局部变量 print(age,name)def f2(): age=19 #局部变量 print(age,name)f1()f2()18 Tim19 Tim 在函数内部也可以定义全局变量: name = 'Tim' #全局变量def f1(): age = 18 #局部变量 global name #定义全局变量 name = 'Eric' print(age,name)f1()print(name) 全局变量默认可读,如果需要改变全局变量的值,需要在函数……

    网友NO.576210

    09-08

    Python学习笔记之函数的定义和作用域实例详解, 本文实例讲述了Python函数的定义和作用域。分享给大家供大家参考,具体如下: 定义函数 默认参数: 可以向函数中添加默认参数,以便为在函数调用中未指定的参数提供默认值 # 如果调用 cylinder_volume 函数时,不提供radius参数,那么radius的值为5def cylinder_volume(height, radius=5): pi = 3.14159 return height * pi * radius ** 2 向函数中的参数传值的方法:按照位置和按照名称 cylinder_volume(10, 7) # 1539.3791cylinder_volume(height=10, radius=7) # 1539.3791cylinder_volume(radius=7, height=10) # 1539.3791 注意:上述第一种是常用的按照位置传值,第二种和第三种是按照名称传值 定义函数[相关练习] 写一个名称为 population_density 的函数,该函数有两个……

    <
    1
    >

    Copyright 2018-2020 www.xz577.com 码农之家

    版权投诉 / 书籍推广 / 赞助:520161757@qq.com