天天看點

Python 内置函數 locals() 和globals()

1》這兩個函數主要提供, 基于字典的通路局部變量和全局變量的方式。

python 使用叫做名字空間的東西來記錄變量的軌迹。 名字空間是一個字典 ,它的鍵就是字元串形式的變量名字,它的值就是變量的實際值。

名字空間可以像 Python 的 dictionary 一樣進行通路。

在一個 Python 程式中的任何一個地方,都存在幾個可用的名字空間。

每個函數都有着自已的名字空間,叫做 局部名字空間,它記錄了函數的變量,包括函數的參數和局部定義的變量。

每個子產品擁有它自已的名字空間,叫做 全局名字空間,它記錄了子產品的變量,包括函數、類、其它導入的子產品、子產品級的變量和常量。

還有就是 内置名字空間, 任何子產品均可通路它,它存放着内置的函數和異常。

2》當一行代碼要使用變量 x 的值時,Python 會到所有可用的名字空間去查找變量,按照如下順序:

局部名字空間 - 特指目前函數或類的方法。如果函數定義了一個局部變量 x, 或一個參數 x,Python 将使用它,然後停止搜尋。

全局名字空間 - 特指目前的子產品。如果子產品定義了一個名為 x 的變量,函數或類,Python 将使用它然後停止搜尋。

内置名字空間 - 對每個子產品都是全局的。作為最後的嘗試,Python 将假設 x 是内置函數或變量。

如果 Python 在這些名字空間找不到 x,它将放棄查找并引發一個 NameError 異常,

同時傳遞 There is no variable named 'x' 這樣一條資訊.

3》名字空間在運作時直接可以通路。局部名字空間可以通過内置的 locals 函數來通路。全局 (子產品級别) 名字空間可以通過内置的 globals 函數來通路。

locals 對局部 (函數) 名字空間做了些什麼,globals 就對全局 (子產品) 名字空間做了什麼。

然而 globals 更令人興奮,因為一個子產品的名字空間包含了子產品級的變量和常量,它還包括了所有在子產品中定義的函數和類,以及任何被導入到子產品中的東西。

4》回想一下 from module import 和 import module 之間的不同?

使用 import module,子產品自身被導入,但是它保持着自已的名字空間,

這就是為什麼您需要使用子產品名來通路它的函數或屬性: module.function 的原因。

但是使用 from module import,實際上是從另一個子產品中将指定的函數和屬性導入到您自己的名字空間,

這就是為什麼您可以直接通路它們卻不需要引用它們所來源的子產品的原因。

使用 globals 函數,您會真切地看到這一切的發生。

5》locals()執行個體:

def foo(arg, a):  
    x = 100 
    y = 'hello python!'  
    for i in range(10):  
        j = 1  
        k = i  
    print locals()  
    
foo(1,2)  
           

結果:

{'a': 2, 'i': 9, 'k': 9, 'j': 1, 'arg': 1, 'y': 'hello python!', 'x': 100}

6》locals 是隻讀的,不可修改, 而globals可以修改,原因是:

locals()實際上沒有傳回局部名字空間,它傳回的是一個拷貝。是以對它進行修改,修改的是拷貝,而對實際的局部名字空間中的變量值并無影響。

globals()傳回的是實際的全局名字空間,而不是一個拷貝: 與 locals 的行為完全相反。

是以對 globals 所傳回的 dictionary 的任何的改動都會直接影響到全局變量的取值。

#!/usr/bin/env python    
#coding:utf-8    
'''This is my first python program!'''    
z = 7 #定義全局變量  
def foo(arg):   
    x = 1   
    print locals()  
    print 'x=',x  
    locals()['x'] = 2 #修改的是局部名字空間的拷貝,而實際的局部名字空間中的變量值并無影響。  
    print locals()  
    print "x=",x  
  
foo(3)   
print globals()  
print 'z=',z  
globals()["z"] = 8 #globals()傳回的是實際的全局名字空間,修改變量z的值  
print globals()  
print "z=",z  
           

結果:

{'x': 1, 'arg': 3}

x= 1

{'x': 1, 'arg': 3}

x= 1

{'foo': <function foo at 0x02A17CF0>, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:\\workspace\\python day03\\main\\test.py', '__package__': None, '__name__': '__main__', 'z': 7, '__doc__': 'This is my first python program!'}

z= 7

{'foo': <function foo at 0x02A17CF0>, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:\\workspace\\python day03\\main\\test.py', '__package__': None, '__name__': '__main__', 'z': 8, '__doc__': 'This is my first python program!'}

z= 8

(完)