5.2 Python 名称空间与作用域

2024-06-13 22:52

本文主要是介绍5.2 Python 名称空间与作用域,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 名称空间

1.1 简介
定义一个变量, num = 1, 解释器会申请内存空间存放值1, 将名字num与1的内存地址的绑定关系存在栈区中.名称空间: 是放名称与对象绑定关系的地方, 是对栈区的一种虚拟的划分, 为了避免程序中标识名称冲突, 
有了名称空间之后就可以在栈区中存放相同的名字, 每个命名空间是独立的, 允许出现相同的标识符名称.
1.2 分类
名称空间划分三个大类:
* 1. 内置名称(built-in names): 解释器内置的名称, : 内置函数, 关键字...
* 2. 全局名称(global names):  自定义的名称, 导入的模块, 顶级的名称(没有缩进的叫顶级)..
* 3. 局部名称(local names): 函数()内部的定义的标识符.
1.3 生命周期
内置: Python解释器启动产生, 关闭销毁.
全局: py文件运行产生, 结束销毁.
局部: 调用函数产生, 调用结束销毁.产生顺序: 内置 --> 全局 --> 局部 --> 子局部 --> 子子局部 --> ...
销毁顺序: ... --> 子子局部 --> 子局部 --> 局部 --> 全局 --> 内置
1.4 内置名称
# 内置名称, 内部定义好的名称能够直接使用.
>>> print
<built-in function print>
>>> len
<built-in function len>
>>> print('xx')
xx
>>> len('x')
1
1.5 全局名称
* 类似if, for, with等语句, 不存在自己的作用域, 这些语句中定义的名称时全局名称.
# 自定义名称
num1 = 0
# 定义后全局可以使用
print(num1)# def 关键字定义的名称, 属于顶级的名称.
def func():# 在函数中使用全局名称, 不能做计算后续说明.print(num1)# 调用顶级的名称
func()
# if, for 不存在自己的作用域
if True:num3 = 3print(num3)  # 3for i in range(5):# 以最后一次为准num4 = i# 这里提示num3没有定义忽略即可, 在运行for循环后就会定义, Pycharm没法检测.
print(num4)  # 0
1.6 局部名称
def func():print('局部空间')# 局部变量num1 = 1print(num1)  # 1if True:# 局部变量num2 = 2print(num2)  # 2for i in range(4):# 局部变量num3 = i# 提示没有定义, 忽略即可, 在执行for循环后会定义.print(num3)  # 3# 局部变量(函数可以嵌套)def func4():...print(func4)func()

2. 作用域

作用域: 名称空间的作用范围, 决定了程序中哪些位置可以什么名称.
按照名称空间中名称的作用范围可以将三种名称空间划分为两个区域:
* 1.全局作用域: 全局作用域中的名称在整个文件执行过程中都不会被销毁, 在任意位置都可以使用.内置名称空间, 全局名称空间属于全局作用域, 全局存活, 全局有效
* 2.局部作用域: 局部作用域中的名称在函数调用时产生, 调用结束时销毁, 只能在函数内使用.局部名称空间属于局部作用域, 临时存活, 局部有效.
* 名称的作用域觉
2.1 名字查找优先级
查找准则: 函数的嵌套关系与名字的查找顺序是在定义阶段就已经确定好的与调用位置无关.
Python在查找"名称", 是按照LEGB规则查找的:
L(Local)     指的是所有函数内部的函数.
E(Enclosing) 指的是最外层函数.
G(Global)    指的是全局变量.
B(Built-in)  指的是Python内置保留的名称.明确自己当前在哪个名称空间中, 再基于当前位置向外查找, 都没有找到就报错: 该名称没有定义
1. LEGB
len = 1def func1():len = 2def func2():len = 3print(len)  # 在当前的名称空间中拿lenfunc2()func1()
2. EGB
len = 1def func1():len = 2def func2():print(len)  # 向外层的名称空间中拿lenfunc2()func1()
3. GB
len = 1def func1():def func2():print(len)  # 向全局的名称空间中拿lenfunc2()func1()
4. B
def func1():def func2():print(len)  # 向内置的名称空间中拿lenfunc2()func1()
2.2 注意事项
* 1. 在局部内可以直接使用全部变量, 但是这个变量不能做数学运算, 如果做运算, 会报错: 名称没有定义.
因为该变量会被Python解释器认为是局部变量而非全局变量.
count1 = 10def index():print(count1)index()count2 = 10def index():print(count + 1)  # NameError: name 'count' is not definedindex()
* 2. 嵌套函数中, 先使用外层函数中的名称, 在定义同样的名称就会报错.
: UnboundLocalError: local variable 'len' referenced before assignment.
: 局部变量'xxx'在赋值之前引用.
def func1():num = 2def func2():# 调用外层的名称, 正确运行.print(num + 1)  # 2func2()func1()
def func1():num = 2def func2():# 想在这个调用外层的len, 报错: ↓print(num + 1)# 使用外层的名称, 往回就不能在此空间定义同样的名称.# UnboundLocalError: local variable 'num' referenced before assignmennum = 3func2()func1()
2.3 修改可变类型值
在不同的名称空间, 修改不可变类型的值, 互不影响. 
num = 1def func1():# 在定义阶段就确定查找关系print(num)  # 1func1()def func2():# 局部的num作用于func2的空间, 不会影响全局的num.num = 2  print(num)  # 2func1()func2()
运行工具窗口显示:
1
2
1
2.4 修改不可变类型值
在不同的名称空间, 可以修改可变类型的值, 操作的是同一个对象, 相互影响.
list1 = []def func1():func2()list1.append('b')print(list1)def func2():list1.append('a')func1()
print(list1)
运行工具窗口显示:
['a', 'b']
['a', 'b']
2.5 globals()函数
Python内置globals()函数: 查看全局的名称, 在全局名称空间中使用, 结果是一个字典类型.
num = 1
print(globals(), type(globals()))  # {... 'num': 1} <class 'dict'># 通过globals()定义全局变量
globals()['num2'] = 2
print(globals())  # {... 'num': 1, 'num2': 2}# 通过globals()取值全局变量的值
print(globals().get('num'))  # 1
print(globals().get('num2'))  # 2
2.6 locals()函数
Python内置locals()函数: 查看局部的名称, 在对应的局部名称中使用, 结果是一个字典类型.
局部名称空间可以用很多个, 查个空间就在那个空间内使用.
def func1():b = 1print(locals(), type(locals()))  # {'b': 1} <class 'dict'>def func2():c = 2print(locals())  # {'c': 2}func2()# 通过locals()定义当层局部变量locals()['b2'] = 3# 通过locals()获取当层局部变量的值print(locals()['b'])  # 1print(locals()['b2'])  # 3func1()
2.7 global关键字
global关键字: 在局部空间中设置声明一个全局变量.
在局部修改全局变量的值:
不可变数据类型需要global声明, 不可变类型的值修改都需要通过赋值操作.
可变类型数据类型则不需要声明, 可变类型的值本身是一个容器, 可以通过方法操作, 清空在插入值都是都可的.
# 修改不可变类型
x = 0def func1():# 在局部空间中设置声明一个全局变量x.global x# 为全局变量x设置值.x = 10func1()
print(x)  # 10
#  修改不可变类型
list1 = []  # 可变类型def func1():# 通过方法可以直接修改list1.append(1)func1()
print(list1)  # [1]
2.8 nonlocal关键字
nonlocal关键字: 在内层的局部空间中声明一个外层的局部空间的名称, 外层是全局就会报错.
仅不可变类型需要nonlocal声明.
def func1():x = 0def func2():nonlocal x  # 在内部局部空间中设置声明一个外部局部空间的名称.x = 10  # 修改x的值.func2()print(x)  # 10space = locals()print(space)# {'func2': <function func1.<locals>.func2 at 0x000001B7774B0510>, 'x': 10}func1()
#  修改可变类型
list1 = []def func1():list1.append(1)def func2():list1.append(2)func2()func1()
print(list1)  # [1, 2]

这篇关于5.2 Python 名称空间与作用域的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1058654

相关文章

Python 字符串占位

在Python中,可以使用字符串的格式化方法来实现字符串的占位。常见的方法有百分号操作符 % 以及 str.format() 方法 百分号操作符 % name = "张三"age = 20message = "我叫%s,今年%d岁。" % (name, age)print(message) # 我叫张三,今年20岁。 str.format() 方法 name = "张三"age

一道经典Python程序样例带你飞速掌握Python的字典和列表

Python中的列表(list)和字典(dict)是两种常用的数据结构,它们在数据组织和存储方面有很大的不同。 列表(List) 列表是Python中的一种有序集合,可以随时添加和删除其中的元素。列表中的元素可以是任何数据类型,包括数字、字符串、其他列表等。列表使用方括号[]表示,元素之间用逗号,分隔。 定义和使用 # 定义一个列表 fruits = ['apple', 'banana

Python应用开发——30天学习Streamlit Python包进行APP的构建(9)

st.area_chart 显示区域图。 这是围绕 st.altair_chart 的语法糖。主要区别在于该命令使用数据自身的列和指数来计算图表的 Altair 规格。因此,在许多 "只需绘制此图 "的情况下,该命令更易于使用,但可定制性较差。 如果 st.area_chart 无法正确猜测数据规格,请尝试使用 st.altair_chart 指定所需的图表。 Function signa

python实现最简单循环神经网络(RNNs)

Recurrent Neural Networks(RNNs) 的模型: 上图中红色部分是输入向量。文本、单词、数据都是输入,在网络里都以向量的形式进行表示。 绿色部分是隐藏向量。是加工处理过程。 蓝色部分是输出向量。 python代码表示如下: rnn = RNN()y = rnn.step(x) # x为输入向量,y为输出向量 RNNs神经网络由神经元组成, python

python 喷泉码

因为要完成毕业设计,毕业设计做的是数据分发与传输的东西。在网络中数据容易丢失,所以我用fountain code做所发送数据包的数据恢复。fountain code属于有限域编码的一部分,有很广泛的应用。 我们日常生活中使用的二维码,就用到foutain code做数据恢复。你遮住二维码的四分之一,用手机的相机也照样能识别。你遮住的四分之一就相当于丢失的数据包。 为了实现并理解foutain

python 点滴学

1 python 里面tuple是无法改变的 tuple = (1,),计算tuple里面只有一个元素,也要加上逗号 2  1 毕业论文改 2 leetcode第一题做出来

Python爬虫-贝壳新房

前言 本文是该专栏的第32篇,后面会持续分享python爬虫干货知识,记得关注。 本文以某房网为例,如下图所示,采集对应城市的新房房源数据。具体实现思路和详细逻辑,笔者将在正文结合完整代码进行详细介绍。接下来,跟着笔者直接往下看正文详细内容。(附带完整代码) 正文 地址:aHR0cHM6Ly93aC5mYW5nLmtlLmNvbS9sb3VwYW4v 目标:采集对应城市的

python 在pycharm下能导入外面的模块,到terminal下就不能导入

项目结构如下,在ic2ctw.py 中导入util,在pycharm下不报错,但是到terminal下运行报错  File "deal_data/ic2ctw.py", line 3, in <module>     import util 解决方案: 暂时方案:在终端下:export PYTHONPATH=/Users/fujingling/PycharmProjects/PSENe

将一维机械振动信号构造为训练集和测试集(Python)

从如下链接中下载轴承数据集。 https://www.sciencedirect.com/science/article/pii/S2352340918314124 import numpy as npimport scipy.io as sioimport matplotlib.pyplot as pltimport statistics as statsimport pandas

Python利用qq邮箱发送通知邮件(已封装成model)

因为经常喜欢写一些脚本、爬虫之类的东西,有需要通知的时候,总是苦于没有太好的通知方式,虽然邮件相对于微信、短信来说,接收性差了一些,但毕竟免费,而且支持html直接渲染,所以,折腾了一个可以直接使用的sendemail模块。这里主要应用的是QQ发邮件,微信关注QQ邮箱后,也可以实时的接收到消息,肾好! 好了,废话不多说,直接上代码。 # encoding: utf-8import lo