Python之作用域解析

作用域:程序创建、访问、改变一个变量时,都是在一个保存该变量的空间内进行,这个空间为命名空间,即作用域。

  • python作用域是静态的,变量被赋值、创建的位置决定了其被访问的范围,即变量作用域由其所在位置决定。

    a = 1 #a为全局变量
    def local(): #local也在全局作用域中
        b = 2 #b为局部变量
    
  • 在Python中,使用一个变量时并不严格要求需要预先声明它,但是在真正使用它之前,它必须被绑定到某个内存对象(被定义、赋值);这种变量名的绑定将在当前作用域中引入新的变量,同时屏蔽外层作用域中的同名变量。

    a = 1
    def local():
        a = 2 #由于python不需要预先声明,因此在局部作用域中引入新的变量,而没有修改全局变量
    local()
    print(a)
    输出1
    
LEGB作用域
  • Local(局部):在函数与类中,每当调用函数时都会创建一个局部作用域,局部变量域像一个栈,仅仅是暂时的存在,依赖创建该局部作用域的函数是否处于活动的状态;
    默认情况下局部是无法修改全局变量的:
    a = [1]
    def local():
       a = [1,2]
    local()
    print(a)
    输出[1]
    
    想要实现局部修改全局变量,有两种办法:
     #(1)增加global,nonlocal关键字
     a = [1]
    def local():
        a = [1,2]
    local()
    print(a)
    输出[1, 2]
    #(2)对于可变对象,如list,dict等,使用内置函数
    b = [1]
    def local():
        b.append(2)
    local()
    print(b)
    输出[1, 2]
    
  • Enclosed(嵌套):一般出现在函数中嵌套了一个函数,在外围的函数中的作用域;主要目的是实现闭包
    def local(i):
       a = [] #此处a在add函数的嵌套作用域内
       def add():
           a.append(i)
           return a
       return add
    temp = local(1)
    print(temp())
    输出[1]
    
  • Global(全局):模块文件顶层声明的变量具有全局作用域,从外部开来,模块的全局变量就是一个模块对象的属性;仅限于单个模块文件中;
  • Built-in(内置):系统内解释器定义的变量,如预定义在builtin 模块内的变量;解释器在则在,解释器亡则亡;

使用规则: 从上往下顺序创建,从下往上搜索,即搜索遵循LEGB原则,如果一直搜索不到则报错。
Python之作用域解析_第1张图片

金句

  • 无声明的情况下,赋值即私有,若外部有相同名称的变量则被遮挡
  • 想修改外部变量,需声明(global、nonlocal),或者通过可变对象的内置函数
  • python2中没有nonlocal关键字,只能用可变对象来临时解决中间层变量修改的问题

注意:Python与C有着很大的区别,在Python中并不是所有的语句块中都会产生作用域。只有当变量在Module(模块)、Class(类)、def(函数)中定义的时候,才会有作用域的概念。如下边语句在python中是不报错的,而在C++等中则报错。

>>>for i in range(3):
...    a = i
>>>print(a)
2

一个易错的例子:

  name = "lbj"
  def f1():
      print(name)
  def f2():
      name = "coldplay"
      f1()
  f2()
  输出lbj
  #在调用f1时,name在局部没有,会去搜索全局而f2的name只是局部的,与全局无关,因此输出lbj

你可能感兴趣的:(Python)