zoukankan      html  css  js  c++  java
  • python 闭包变量不允许write,要使用nonlocal

    以下是一段简单的闭包代码示例:

    def foo():
        m=3
        n=5
        def bar():
            a=4
            return m+n+a
        return bar
    
    >>>bar =  foo()
    >>>bar()
    12
    

    是可以的!但是:
    def foo():
        m=3
        n=5
        def bar():
            a=4
    m += 1 #不可以!!! return m+n+a return bar >>>bar = foo() >>>bar() 12

    
    

    python中global 和 nonlocal 的作用域

    
    

    python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量 。

    
    

    一 global

    
    

    global关键字用来在函数或其他局部作用域中使用全局变量。但是如果不修改全局变量也可以不使用global关键字。

    
    
    复制代码
    1 gcount = 0
    2 
    3 def global_test():
    4     gcount+=1
    5     print (gcount)
    6 global_test()
    复制代码
    
    

    D:Python34python.exe E:/PycharmProjects/Day3/globaltest.py
    Traceback (most recent call last):
    File "E:/PycharmProjects/Day3/globaltest.py", line 6, in <module>
    global_test()
    File "E:/PycharmProjects/Day3/globaltest.py", line 4, in global_test
    gcount+=1
    UnboundLocalError: local variable 'gcount' referenced before assignment

    
    

    Process finished with exit code 1

    
    

    第一行定义了一个全局变量,(可以省略global关键字)。

    
    

    在global_test 函数中程序会因为“如果内部函数有引用外部函数的同名变量或者全局变量,并且对这个变量有修改.那么python会认为它是一个局部变量,又因为函数中没有gcount的定义和赋值,所以报错。

    
    
    
    
    

    二、声明全局变量,如果在局部要对全局变量修改,需要在局部也要先声明该全局变量:

    
    
    1
    2
    3
    4
    5
    6
    7
    gcount = 0
     
    def global_test():
        global  gcount
        gcount+=1
        print (gcount)
    global_test()
    
    

      如果在函数中声明 gcount 是全局变量,即可对其进行修改。 正确输出 1 。

    
    
    
    
    

    三、 在局部如果不声明全局变量,并且不修改全局变量。则可以正常使用全局变量:

    
    
    1
    2
    3
    4
    5
    gcount = 0
     
    def global_test():
        print (gcount)
    global_test()
    
    

      如果在局部不修改全局变量,程序正确输出 0 。

    
    
    
    
    

    四、nonlocal关键字用来在函数或其他作用域中使用外层(非全局)变量。

    
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    def make_counter(): 
        count = 0 
        def counter(): 
            nonlocal count 
            count += 1 
            return count 
        return counter 
           
    def make_counter_test(): 
      mc = make_counter() 
      print(mc())
      print(mc())
      print(mc())
     
    make_counter_test()
    
    

      输出:

    
    

       1

    
    

       2

    
    

       3

    
    

    五、

    
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    def scope_test():
        def do_local():
            spam = "local spam" #此函数定义了另外的一个spam字符串变量,并且生命周期只在此函数内。此处的spam和外层的spam是两个变量,如果写出spam = spam + “local spam” 会报错
        def do_nonlocal():
            nonlocal  spam        #使用外层的spam变量
            spam = "nonlocal spam"
        def do_global():
            global spam
            spam = "global spam"
        spam = "test spam"
        do_local()
        print("After local assignmane:", spam)
        do_nonlocal()
        print("After nonlocal assignment:",spam)
        do_global()
        print("After global assignment:",spam)
     
    scope_test()
    print("In global scope:",spam)
    
    

      输出是:

    
    

    After local assignmane: test spam
    After nonlocal assignment: nonlocal spam
    After global assignment: nonlocal spam
    In global scope: global spam

    
    
    
    
    
    
    
    

    在函数 add_b 内 global 定义的变量 b,只能在 函数 do_global 内引用, 如果要在 do_global 内修改,必须在 do_global 函数里面声明 global  b ,表明是修改外面的 全局变量 b :

    
    
    复制代码
    def add_b():
        global  b
        b = 42
        def do_global():
            global  b
            b = b + 10
            print(b)
        do_global()
        print(b)
    add_b()
    复制代码
    
    
    
    
    

    global 定义的变量,表明其作用域在局部以外,即局部函数执行完之后,不销毁 函数内部以global定义的变量:

    
    
    def add_a():
        global  a
        a =   3
    add_a()
    print(a)   
    
    

    输出 3 。

    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    
    复制代码
    def add_b():
        global  b
        b = 42
        def do_global():
            global  b
            b = b + 10
            print(b)
        do_global()
        print(b)
    add_b()
    print(b)
    复制代码
    
    

    以上代码输出:

    
    

    52

    
    

    52

    
    

    52

    
    
    
    
    
    复制代码
    def add_b():
        global  b
        b = 42
        def do_global():
            #global  b
            b = b + 10
            print(b)
        do_global()
        print(b)
    add_b()
    复制代码
    
    

    以上代码报错:

    
    

    Traceback (most recent call last):
    File "E:/PycharmProjects/OOP/exe1.py", line 42, in <module>
    add_b()
    File "E:/PycharmProjects/OOP/exe1.py", line 40, in add_b
    do_global()
    File "E:/PycharmProjects/OOP/exe1.py", line 38, in do_global
    b = b + 10
    UnboundLocalError: local variable 'b' referenced before assignment

    
    

    原因: global 定义的 b ,只能引用,不能修改。

    
    
    
    
    
    复制代码
    def add_b():
        global  b
        b = 42
        def do_global():
            global  a
            a = b + 10
            print(b)
        do_global()
        print(a)
    add_b()
    print("a = %s , b = %s " %(a, b))
    复制代码
    
    

    输出:

    
    

    42
    52
    a = 52 , b = 42

    
    
    
    
    
    
    
    
    
    
    
    复制代码
    def add_b():
        #global  b
        b = 42
        def do_global():
            global  b
            b =  10
            print(b)
        do_global()
        print(b)
    add_b()
    print(" b = %s " % b)
    复制代码
    
    

    以上代码输出:

    
    

    10
    42
    b = 10

    
    
    
    
    
    
    
    
    复制代码
    def add_b():
        #global  b
        b = 42
        def do_global():
            nonlocal  b
            b =  10
            print(b)
        do_global()
        print(b)
    add_b()
    复制代码
    
    

    以上代码输出:

    
    

    10

    
    

    10

    
    
    
    
    
    
    
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    def add_b():
        #global  b
        b = 42
        def do_global():
            nonlocal  b
            b =  10
            print(b)
        do_global()
        print(b)
    add_b()
    print(" b = %s " % b)
    
    

      以上代码报错:

    
    

    print(" b = %s " % b)
    NameError: name 'b' is not defined

    
    

    说明: nonlocal  适用于在局部函数 中 的局部函数, 把最内层的局部 变量设置成外层局部可用,但是还不是全局的。

    
    
    
    
    
    复制代码
    def add_b():
        #global  b
        #b = 42
        def do_global():
            nonlocal  b
            b =  10
            print(b)
        do_global()
        #print(b)
    add_b()
    复制代码
    
    

    以上代码报错:

    
    

    File "E:/PycharmProjects/OOP/exe1.py", line 37
    nonlocal b
    SyntaxError: no binding for nonlocal 'b' found

    
    

    nonlocal 要绑定一个局部变量。

     
  • 相关阅读:
    javascript规范
    freemarker规范
    java代码质量
    使用ESP8266制作一个微型气象站
    热风枪焊接表面贴装元件的工具和技巧
    MCUXpresso IDE:导入Kinetis Design Studio工程
    基于LPCXpresso54608开发板创建Embedded Wizard UI应用程序
    STM32 LoRaWAN探索板B-L072Z-LRWAN1入门指南
    LPCXpresso54608开发板中文用户手册
    STM32 LoRaWAN探索板B-L072Z-LRWAN1中文用户手册
  • 原文地址:https://www.cnblogs.com/bonelee/p/8641621.html
Copyright © 2011-2022 走看看