深入了解python基础, 用最基础的知识吊打面试官。
在Python程序中创建、改变、查找变量名时,都是在一个保存变量名的空间中进行,我们称之为命名空间
,也被称之为作用域
。
Python的作用域是静态的
,在源代码中变量名被赋值的位置决定了该变量能被访问的范围
。即Python变量的作用域由变量所在源代码中的位置
决定。
只有当变量在Module(模块)、Class(类)、def(函数)中定义的时候,才会有作用域的概念。
在作用域中定义的变量,一般只在该作用域中有效。
在if-elif-else
、for-else
、while
、try-except / try-finally
等关键字的语句块中并不会产生作用域
。
我们首先在函数中定义一个变量number:
def func():
number = 100
print(number)
print(number)
执行函数我们会得到一个错误ERROR: NameError: name ‘number’ is not defined
上面是函数, 我们在定义一个条件内定义number的代码:
if True:
number = 100
print(number)
print("******输出分隔符******")
print(number)
执行这段代码得到的结果:
由此可以得出结论:
在我们的def函数
中是存在作用域的, 而在我们的if
条件demo 中是不存在作用域的。
L(local):局部作用域(函数内的命名空间)
E(enclosing)嵌套作用域 (外部嵌套函数的命名空间)
G(global)全局作用域(所在模块(文件)的命名空间)
B(built-in)内置作用域(Python内置模块的命名空间)
globalVar = 100 # 全局作用域 (既然是全局就代表任何地方可用)
def nest_scope():
enclosingVar = 200 # 嵌套作用域(可在嵌套函数中使用)
def func():
localVar = enclosingVar + 1 # 局部作用域(只能在当前函数func 使用)
print(localVar)
return func()
print(__name__) # 内置作用域
当在函数中使用未确定的变量名时,Python会按照优先级依次搜索4个作用域,以此来确定该变量名的意义。
首先搜索局部作用域(L),
之后是上一层嵌套结构中def或lambda函数的嵌套作用域(E),
之后是全局作用域(G),
最后是内置作用域(B)。
按这个查找原则,在第一处找到的地方停止。如果没有找到,则会出发NameError
错误。
局部作用域
> 嵌套作用域
> 全局作用域
> 内置作用域
|-------------------------------------示例1--------------------------------------|
def func():
number = 256
print(number) # 输出局部作用域的number
number = 1024
func()
print(number) # 输出全局作用域的number
运行结果:
256
1024
|-------------------------------------示例2--------------------------------------|
def nest_scope():
number = 256
print(number)
def func():
print(number) # 此处输出的是上层嵌套函数nest_scope()中的number
func()
number = 1024
nest_scope()
print(number)
运行结果:
256
256
1024
|-------------------------------------示例3--------------------------------------|
number = 1024
def func():
print(number) # 此处的变量number对应的肯定是下一行中所声明的局部变量, 但是未被赋值之前引用了所以会报错
number = 256
func()
print(number)
运行结果:
UnboundLocalError: local variable 'number' referenced before assignment
|-------------------------------------示例4--------------------------------------|
number = 1024
def func():
print(number)
# number = 256 # 此行定义的局部变量被注释掉之后, 上一行所对应的应该是全局变量中的number
func()
print(number)
运行结果:
1024
1024
以上代码可清晰的得出结论LEGB法则的优先顺序。
number = 1024
def test_update_number():
def nested_func():
global number # 绑定到了第一行定义的number
print('current_number=', number) # 输出当前number的值
number = 200
return nested_func()
test_update_number()
print(number) # 输出修改后的值
输出结果:
current_number= 1024
200
def func_outer():
number = 10
def func_inner():
nonlocal number # 绑定到了第二行定义的number
print("current_number=", number) # 未被修改之前的值
number = 20
print(number) # 内层输出修改后的值
func_inner()
print(number) # 外层输出修改后的值
func_outer()
输出结果:
current_number= 10
20
20
由此可看出各自的不同,以及正确的用法。
基础知识的整理也是相当重要的, 日常开发过程中,以及面试过程中都会被用到,好了今日分享已结束, 欢迎各位大佬指正。