開發與維運

作用域|Python從入門到精通:高階篇之九

文檔字符串 | Python從入門到精通:高階篇之八

作用域

作用域指的是變量生效的區域。
示例代碼:

b = 20 # 全局變量

def fn():
    a = 10 # a定義在了函數內部,所以他的作用域就是函數內部,函數外部無法訪問
    print('函數內部:','a =',a)
    print('函數內部:','b =',b)

fn()

# print('函數外部:','a =',a)
print('函數外部:','b =',b)

執行結果:

image.png

在Python中一共有兩種作用域:

  1. 全局作用域

    • 全局作用域在程序執行時創建,在程序執行結束時銷燬
    • 所有函數以外的區域都是全局作用域
    • 在全局作用域中定義的變量,都屬於全局變量,全局變量可以在程序的任意位置被訪問
  2. 函數作用域

    • 函數作用域在函數調用時創建,在調用結束時銷燬
    • 函數每調用一次就會產生一個新的函數作用域
    • 在函數作用域中定義的變量,都是局部變量,它只能在函數內部被訪問

通過運行如下代碼我們可以直觀的瞭解到作用域的概念:

b = 20 # 全局變量

def fn():
    a = 10 # a定義在了函數內部,所以他的作用域就是函數內部,函數外部無法訪問
    print('函數內部:','a =',a)
    print('函數內部:','b =',b)
fn()

# print('函數外部:','a =',a)
print('函數外部:','b =',b)

def fn2():
    def fn3():
        print('fn3中:','a =',a)
    fn3()
fn2()

執行結果:

image.png

變量的查找:
當我們使用變量時,會優先在當前作用域中尋找該變量:

  • 如果有則使用
  • 如果沒有則繼續去上一級作用域中尋找,如果有則使用,運行代碼如下:
a = 10
def fn2():
    def fn3():
        print('fn3中:','a =',a)
    fn3()
fn2()

執行結果:

image.png

  • 如果依然沒有則繼續去上一級作用域中尋找,以此類推
  • 直到找到全局作用域,依然沒有找到,則會拋出異常:NameError: name 'a' is not defined,運行代碼如下:
def fn2():
    def fn3():
        print('fn3中:','a =',a)
    fn3()
fn2()

執行結果:

image.png

若在外層再定義一個fn3,它和剛才的fn3有什麼關係呢?

def fn2():
    def fn3():
        print('fn3中:','a =',a)
    fn3()

# fn2()

a = 20

def fn3():
    a = 10
    print('a = ',a)

fn3()

執行結果:

image.png

若將fn3中的a去掉,代碼再次運行,執行結果:

image.png

在fn3中不聲明新的a變量,直接寫a=10,會怎麼樣呢?

def fn2():
    def fn3():
        print('fn3中:','a =',a)
    fn3()

# fn2()

a = 20

def fn3():
    # a = 10 # 在函數中為變量賦值時,默認都是為局部變量賦值
    a = 10
    print('函數內部:','a =',a)

fn3()
print('函數外部:','a =',a)

執行結果:

image.png

如果希望在函數內部修改全局變量,則需要使用global關鍵字,來聲明變量,代碼如下:

def fn2():
    def fn3():
        print('fn3中:','a =',a)
    fn3()

# fn2()

a = 20

def fn3():
    # a = 10 # 在函數中為變量賦值時,默認都是為局部變量賦值
    global a # 聲明在函數內部的使用a是全局變量,此時再去修改a時,就是在修改全局的a
    a = 10 # 修改全局變量
    print('函數內部:','a =',a)

fn3()
print('函數外部:','a =',a)

執行結果:

image.png

配套視頻課程,點擊這裡查看

獲取更多資源請訂閱Python學習站

Leave a Reply

Your email address will not be published. Required fields are marked *