Python 变量作用域的规则是 LEGB

上篇 https://www.cnblogs.com/yvivid/p/python_LEGB_1.html

下篇 https://www.cnblogs.com/yvivid/p/python_LEGB_2.html

LEGB含义解释:
L —— Local(function);函数内的名字空间
E —— Enclosing function locals;外部嵌套函数的名字空间(例如closure)
G —— Global(module);函数定义所在模块(文件)的名字空间
B —— Builtin(Python);Python内置模块的名字空间

一、Builtin

该部分其实主要是 Python 自带的 内置命名空间,主要是 内置函数,异常类 等。可以通过 dir(__builtins__) 来查看:

>>> dir(__builtins__)
['ArithmeticError', 'AssertionError', 'AttributeError', 'BaseException', 'BlockingIOError', 'BrokenPipeError',
'BufferError', 'BytesWarning', 'ChildProcessError', 'ConnectionAbortedError', 'ConnectionError',
'ConnectionRefusedError', 'ConnectionResetError', 'DeprecationWarning', 'EOFError', 'Ellipsis', 'EnvironmentError',
'Exception', 'False', 'FileExistsError', 'FileNotFoundError', 'FloatingPointError', 'FutureWarning',
'GeneratorExit', 'IOError', 'ImportError',
'ImportWarning', 'IndentationError', 'IndexError', 'InterruptedError',
'IsADirectoryError', 'KeyError', 'KeyboardInterrupt', 'LookupError', 'MemoryError', 'NameError', 'None',
'NotADirectoryError', 'NotImplemented', 'NotImplementedError', 'OSError', 'OverflowError',
'PendingDeprecationWarning', 'PermissionError', 'ProcessLookupError', 'ReferenceError', 'ResourceWarning',
'RuntimeError', 'RuntimeWarning', 'StopIteration', 'SyntaxError', 'SyntaxWarning', 'SystemError', 'SystemExit',
'TabError', 'TimeoutError', 'True', 'TypeError', 'UnboundLocalError', 'UnicodeDecodeError', 'UnicodeEncodeError',
'UnicodeError', 'UnicodeTranslateError', 'UnicodeWarning', 'UserWarning', 'ValueError', 'Warning', 'WindowsError',
'ZeroDivisionError', '_', '__build_class__', '__debug__', '__doc__', '__import__', '__loader__', '__name__',
'__package__', '__spec__', 'abs', 'all', 'any', 'ascii', 'bin', 'bool', 'bytearray', 'bytes', 'callable', 'chr',
'classmethod', 'compile', 'complex', 'copyright', 'credits', 'delattr', 'dict', 'dir', 'divmod', 'enumerate',
'eval', 'exec', 'exit', 'filter', 'float', 'format', 'frozenset', 'getattr', 'globals', 'hasattr', 'hash', 'help',
'hex', 'id', 'input', 'int', 'isinstance', 'issubclass', 'iter', 'len', 'license', 'list', 'locals', 'map', 'max',
'memoryview', 'min', 'next', 'object', 'oct', 'open', 'ord', 'pow', 'print', 'property', 'quit', 'range', 'repr',
'reversed', 'round', 'set', 'setattr', 'slice', 'sorted', 'staticmethod', 'str', 'sum', 'super', 'tuple', 'type',
'vars', 'zip']

该部分的部分一般情况不应该修改。

1)真正修改/删除 或 新增时,需要 __builtins__.???  来指定。

2)另外,你可以在本地增加同名的命名,由于 Locals 和 Global 优先于 builtin,该命名的优先级就高于 Builtin 了。

例如,yvivid 在 Global下 str定义为123,调用 str时,就会先调用 Global 下的 str(即 123)

>>> str = 123
>>> print(str)
123
>>> dir()
['__builtins__', '__doc__', '__loader__', '__name__', '__package__', '__spec__', 'str']

二、Global

该部分就是 本 Module层,如果 在 交互界面,就是当前交互界面的场景。

通过 globals() 可以查看 globals 的所有变量。

>>> globals()
{'__doc__': None, '__spec__': None, '__loader__': <class '_frozen_importlib.BuiltinImporter'>, '__builtins__': <module 'builtins' (built-in)>, '__package__': None, '__name__': '__main__'}

调用 模块时,就需要指定路径,如下例的 sys.int_info

>>> import sys
>>> sys.int_info
sys.int_info(bits_per_digit=30, sizeof_digit=4)

三、Local

该部分 一般指 函数内 的变量空间,通过 locals() 可以查看 Local 的所有变量。

1)直接在 module 层,或 交互界面 使用 locals() 其返回结果同 globals完全相同。因为,没有一个独立的local场景,global 和 local 就是同一个命名空间。

>>> globals()
{'__doc__': None, '__spec__': None, '__loader__': <class '_frozen_importlib.BuiltinImporter'>, '__builtins__': <module 'builtins' (built-in)>, '__package__': None, '__name__': '__main__'}
>>> locals()
{'__doc__': None, '__spec__': None, '__loader__': <class '_frozen_importlib.BuiltinImporter'>, '__builtins__': <module 'builtins' (built-in)>, '__package__': None, '__name__': '__main__'}

2)在函数中,查看locals。如下例:

>>> x = 10
>>> def func(y=3):
temp = 7
print('Locals =', locals())
print('Globals =', globals())
return y+temp
>>> t = func()
Locals = {'temp': 7, 'y': 3}
Globals = {'__loader__': <class '_frozen_importlib.BuiltinImporter'>, '__package__': None, 'x': 10, '__spec__': None, '__name__': '__main__',
'__builtins__': <module 'builtins' (built-in)>, 'func': <function func at 0x0000000003E5B378>, '__doc__': None}

可以看到 locals() 和 globals() 的命名空间 展现出 较大的差异。

其他注意要点:

1)for 循环不是 local 的: 但 列表推导时,作用域 local 的。---- yvivid:重点提醒,这个要关注,后面 Enclose 的部分还会讲到。

>>> for i in range(4):
print(locals())
.... # 重复了四遍的 同 globals() 相关的命名 空间
>>> print(i)
3 # i 依然比记录下来

>>> i = -1
>>> for i in range(3):
        print(i)
        i = 77  # 循环内,对于上一句print(i) 无法生效,因此,被 for i in range(3)重新覆盖了
0
1
2
>>> print(i)   # 最终值 时 i=77 决定。
77

>>> [ locals() for i in range(4)]
[{'.0': <range_iterator object at 0x00000000036AA7F0>, 'i': 3},
{'.0': <range_iterator object at 0x00000000036AA7F0>, 'i': 3},
{'.0': <range_iterator object at 0x00000000036AA7F0>, 'i': 3},
{'.0': <range_iterator object at 0x00000000036AA7F0>, 'i': 3}]
>>> print(i)
Traceback (most recent call last):
File "<pyshell#1>", line 1, in <module>
print(i)
NameError: name 'i' is not defined

 2)在函数中,使用Global 是个坏习惯。这个不多说了,大家基本能够理解。

下篇 https://www.cnblogs.com/yvivid/p/python_LEGB_2.html

说明 Enclosing function locals;外部嵌套函数的名字空间(例如closure) 的作用域。

-------

A、代码 都基于 Python 3.4 环境。

B、参考书籍 《Python学习手册(第四版)》

【原创文档,引用请声明出处,yvivid】https://www.cnblogs.com/yvivid/p/python_LEGB_1.html

Python 变量作用域 LEGB (上)—— Local,Global,Builtin的更多相关文章

  1. Python 变量作用域 LEGB (下)—— Enclosing function locals

    上篇:Python 变量作用域 LEGB (上)—— Local,Global,Builtin https://www.cnblogs.com/yvivid/p/python_LEGB_1.html ...

  2. Python 变量作用域与函数

    Python 的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间,吉多·范罗苏姆为了在阿姆斯特丹打发时间,决心开发一个新的脚本解释程序,作为ABC语言的一种继承.Py ...

  3. python变量作用域

    [python变量作用域] 几个概念: python能够改变变量作用域的代码段是def.class.lamda. if/elif/else.try/except/finally.for/while 并 ...

  4. Python基本语法_变量作用域LEGB

    目录 目录 软件系统 变量的作用域 高级语言对数据类型的使用过程 作用域的产生 作用域的类型 Llocal局部作用域 Eenclosing嵌套作用域 Gglobal全局作用域 Bbuilt-in内置作 ...

  5. python——变量作用域及嵌套作用域

    ----------------------------------------------------------------------------- 前言-------------------- ...

  6. python函数作用域LEGB

    我们的在学习Python函数的时候,经常会遇到很多定义域的问题,全部变量,内部变量,内部嵌入的函数,等等,Python是如何查找的呢?以及Python又是按照什么顺序来查找的呢?这里做一个顺序的说明 ...

  7. python 变量作用域、闭包

    先看一个问题: 下面代码输出的结果是0,换句话说,这个fucn2虽然已经用global声明了variable1,但还是没有改变变量的值 def func1(): variable1=0 def fun ...

  8. python变量作用域,函数与传参

    一.元组传值: 一般情况下函数传递参数是1对1,这里x,y是2个参数,按道理要传2个参数,如果直接传递元祖,其实是传递一个参数 >>> def show( x, y ): ... p ...

  9. Python变量作用域(一)

    在一个程序中使用变量名时,Python创建.改变或者查找变量名都是在所谓的命名空间中进行的.作用域指的就是命名空间. Python中的变量名在第一次赋值时已经创建,并且必须经过赋值后才能够使用.由于变 ...

随机推荐

  1. oracle rollback 观察时间

    ###########issue 0: db alert 有如下提示, thread 1 cannot allocatete new log, sequenec 1111 通过检查v$log ,发现1 ...

  2. 【持续更新】Java 字符串相关问题

    区别 String s1="xxx" 与 String s2=new String("xxx") 的区别 equals() 和 == 的区别 单引号与双引号的区 ...

  3. 洛谷P3959 宝藏(模拟退火乱搞)

    题意 题目链接 题面好长啊...自己看吧.. Sol 自己想了一个退火的思路,没想到第一次交85,多退了几次就A了哈哈哈 首先把没用的边去掉,然后剩下的边从小到大排序 这样我们就得到了一个选边的序列, ...

  4. vuejs 组件 移动端push 没有渲染页面

    this.idcards.push(arr) 这个无效 就知道了 vuejs有个跟push相同的方法 console.log(this.list.push.toString()) 这个push是个同名 ...

  5. Dom EVENT对象

    Event 对象代表事件的状态,比如事件在其中发生的元素.键盘按键的状态.鼠标的位置.鼠标按钮的状态. 事件通常与函数结合使用,函数不会在事件发生前被执行! 一:测试按键 function which ...

  6. jsp跳转标签<jsp:forward>

    forward.jsp <%@ page language="java" contentType="text/html; charset=utf-8" p ...

  7. 【Python图像特征的音乐序列生成】深度卷积网络,以及网络核心

    这个项目主要涉及到两个网络,其中卷积神经网络用来提取图片表达的情绪,提取出一个二维向量. 网络结构如图: 词向量采用预训练的glove模型,d=50,其他信息包括了图片的“空旷程度”.亮度.对比度等信 ...

  8. 从照片网站pexels批量爬取照片

    调试中,未成功. from bs4 import BeautifulSoup import requests headers={ #'User-Agent':'Nokia6600/1.0 (3.42. ...

  9. Unity中实现全局管理类的几种方式

    (搬运自我在SegmentFault的博客) 如何在Unity中实现全局管理类?由于Unity脚本的运行机制和面向组件编程(COP)的思想,实现起来和普通的方式略有差别. 第一种方式是使用静态类.适合 ...

  10. Netbackup常用命令--bpdbjobs

    1.大纲 bpdbjobs – 与 NetBackup 作业数据库进行交互 bpdbjobs [-report] [-M master_servers] [-ignore_parent_jobs] [ ...