深入理解Python中變量賦值的問(wèn)題
前言
在Python中變量名規(guī)則與其他大多數(shù)高級(jí)語(yǔ)言一樣,都是受C語(yǔ)言影響的,另外變量名是大小寫(xiě)敏感的。
Python是動(dòng)態(tài)類型語(yǔ)言,也就是說(shuō)不需要預(yù)先聲明變量類型,變量的類型和值在賦值那一刻被初始化,下面詳細(xì)介紹了Python的變量賦值問(wèn)題,一起來(lái)學(xué)習(xí)學(xué)習(xí)吧。
我們先看一下如下代碼:
c = {}
def foo():
f = dict(zip(list("abcd"), [1, 2 ,3 ,4]))
c.update(f)
if __name__ == "__main__":
a = b = d = c
b['e'] = 5
d['f'] = 6
foo()
print(a)
print(b)
print(c)
print(d)
輸出結(jié)果:
{'a': 1, 'c': 3, 'b': 2, 'e': 5, 'd': 4, 'f': 6}
{'a': 1, 'c': 3, 'b': 2, 'e': 5, 'd': 4, 'f': 6}
{'a': 1, 'c': 3, 'b': 2, 'e': 5, 'd': 4, 'f': 6}
{'a': 1, 'c': 3, 'b': 2, 'e': 5, 'd': 4, 'f': 6}
如果你對(duì)以上輸出結(jié)果不感到奇怪,那么就不必往下看了。實(shí)際上本文要討論的內(nèi)容非常簡(jiǎn)單,不要為此浪費(fèi)您寶貴的時(shí)間。
Python 屬于動(dòng)態(tài)語(yǔ)言,程序的結(jié)構(gòu)可以在運(yùn)行的過(guò)程中隨時(shí)改變,而且 python 還是弱類型的語(yǔ)言,所以如果你是從靜態(tài)、強(qiáng)類型編程語(yǔ)言轉(zhuǎn)過(guò)來(lái)的,理解起 Python 的賦值,剛開(kāi)始可能會(huì)感覺(jué)有些代碼有點(diǎn)莫名其妙。
可能你會(huì)以為上面代碼的輸出會(huì)是這樣的:
{}
{'e': 5}
{}
{'f': 6}
你可能認(rèn)為 a 沒(méi)有被改變,因?yàn)闆](méi)有看到哪里對(duì)它做了改變;b 和 d 的改變是和明顯的;c 呢,因?yàn)槭窃诤瘮?shù)內(nèi)被改變的,你可能認(rèn)為 c 會(huì)是一個(gè)局部變量,所以全局的 c 不會(huì)被改變。
實(shí)際上,這里的 a, b, c, d 同時(shí)指向了一塊內(nèi)存空間,這可內(nèi)存空間保存的是一個(gè)字典對(duì)象。這有點(diǎn)像 c 語(yǔ)言的指針,a, b, c, d 四個(gè)指針指向同一個(gè)內(nèi)存地址,也就是給這塊內(nèi)存其了 4 個(gè)筆名。所以,不管你改變誰(shuí),其他三個(gè)變量都會(huì)跟著變化。那為什么 c 在函數(shù)內(nèi)部被改變,而且沒(méi)有用 global 申明,但全局的 c 去被改變了呢?
我們?cè)賮?lái)看一個(gè)例子:
>>>a = {1:1, 2:2}
>>>b = a
>>>a[3] = 3
>>>b
{1: 1, 2: 2, 3: 3}
>>>a = 4
>>>b
{1: 1, 2: 2, 3: 3}
>>>a
4
當(dāng) b = a 時(shí),a 與 b 指向同一個(gè)對(duì)象,所以在 a 中添加一個(gè)元素時(shí),b 也發(fā)生變化。而當(dāng) a = 4 時(shí), a 就已經(jīng)不再指向字典對(duì)象了,而是指向一個(gè)新的 int 對(duì)象(python 中整數(shù)也是對(duì)象),這時(shí)只有 b 指向字典,所以 a 改變時(shí) b 沒(méi)有跟著變化。這是只是說(shuō)明了什么時(shí)候賦值變量會(huì)發(fā)生質(zhì)的改變,而以上的問(wèn)題還沒(méi)有被解決。
那么,我么再來(lái)看一個(gè)例子:
class TestObj(object):
pass
x = TestObj()
x.x = 8
d = {"a": 1, "b": 2, "g": x}
xx = d.get("g", None)
xx.x = 10
print("x.x:%s" % x.x)
print("xx.x: %s" % xx.x)
print("d['g'].x: %s" % d['g'].x)
# Out:
# x.x:10
# xx.x: 10
# d['g'].x: 10
由以上的實(shí)例可以了解到,如果僅改變對(duì)象的屬性(或者說(shuō)成是改變結(jié)構(gòu)),所有指向該對(duì)象的變量都會(huì)隨之改變。但是如果一個(gè)變量重新指向了一個(gè)對(duì)象,那么其他指向該對(duì)象的變量不會(huì)隨之變化。所以,最開(kāi)始的例子中,c 雖然在函數(shù)內(nèi)部被改變,但是 c 是全局的變量,我們只是在 c 所指向的內(nèi)存中添加了一個(gè)值,而沒(méi)有將 c 指向另外的變量。
需要注意的是,有人可能會(huì)認(rèn)為上例中的最后一個(gè)輸出應(yīng)該是 d['g'].x: 8。 這樣理解的原因可能是覺(jué)得已經(jīng)把字典中 ‘g' 所對(duì)應(yīng)的值取出來(lái)了,并重新命名為 xx,那么 xx 就與字典無(wú)關(guān)了。其實(shí)際并不是這樣的,字典中的 key 所對(duì)應(yīng)的 value 就像是一個(gè)指針指向了一片內(nèi)存區(qū)域,訪問(wèn)字典中 key 時(shí)就是去該區(qū)域取值,如果將值取出來(lái)賦值給另外一個(gè)變量,例如 xx = d['g'] 或者 xx = d.get("g", None),這樣只是讓 xx 這個(gè)變量也指向了該區(qū)域,也就是說(shuō)字典中的鍵 ‘g' 和 xx 對(duì)象指向了同一片內(nèi)存空間,當(dāng)我們只改變 xx 的屬性時(shí),字典也會(huì)發(fā)生變化。
下例更加直觀的展示了這一點(diǎn):
class TestObj(object):
pass
x = TestObj()
x.x = 8
d = {"a": 1, "b": 2, "g": x}
print(d['g'].x)
xx = d["g"]
xx.x = 10
print(d['g'].x)
xx = 20
print(d['g'].x)
# Out:
# 8
# 10
# 10
這個(gè)知識(shí)點(diǎn)非常簡(jiǎn)單,但如果沒(méi)有理解,可能無(wú)法看明白別人的代碼。這一點(diǎn)有時(shí)候會(huì)給程序設(shè)計(jì)帶來(lái)很大的便利,例如設(shè)計(jì)一個(gè)在整個(gè)程序中保存狀態(tài)的上下文:
class Context(object): pass def foo(context): context.a = 10 context.b = 20 x = 1 def hoo(context): context.c = 30 context.d = 40 x = 1 if __name__ == "__main__": context = Context() x = None foo(context) hoo(context) print(x) print(context.a) print(context.b) print(context.c) print(context.d) # Out: # None # 10 # 20 # 30 # 40
示例中我們可以把需要保存的狀態(tài)添加到 context 中,這樣在整個(gè)程序的運(yùn)行過(guò)程中這些狀態(tài)能夠被任何位置被使用。
在來(lái)一個(gè)終結(jié)的例子,執(zhí)行外部代碼:
outer_code.py
from __future__ import print_function
def initialize(context):
g.a = 333
g.b = 666
context.x = 888
def handle_data(context, data):
g.c = g.a + g.b + context.x + context.y
a = np.array([1, 2, 3, 4, 5, 6])
print("outer space: a is %s" % a)
print("outer space: context is %s" % context)
main_exec.py
from __future__ import print_function
import sys
import imp
from pprint import pprint
class Context(object):
pass
class PersistentState(object):
pass
# Script starts from here
if __name__ == "__main__":
outer_code_moudle = imp.new_module('outer_code')
outer_code_moudle.__file__ = 'outer_code.py'
sys.modules["outer_code"] = outer_code_moudle
outer_code_scope = code_scope = outer_code_moudle.__dict__
head_code = "import numpy as np\nfrom main_exec import PersistentState\ng=PersistentState()"
exec(head_code, code_scope)
origin_global_names = set(code_scope.keys())
with open("outer_code.py", "rb") as f:
outer_code = f.read()
import __future__
code_obj = compile(outer_code, "outer_code.py", "exec", flags=__future__.unicode_literals.compiler_flag)
exec(code_obj, code_scope)
# 去除掉內(nèi)建名字空間的屬性,僅保留外部代碼中添加的屬性
outer_code_global_names = set(outer_code_scope.keys()) - origin_global_names
outer_func_initialize = code_scope.get("initialize", None)
outer_func_handle_data = code_scope.get("handle_data", None)
context = Context()
context.y = 999
outer_func_initialize(context)
outer_func_handle_data(context, None)
g = outer_code_scope["g"]
assert g.c == 2886
print("g.c: %s" % g.c)
print(dir(g))
print(dir(context))
pprint(outer_code_moudle.__dict__)
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流。
相關(guān)文章
使用wxPython實(shí)現(xiàn)Windows11任務(wù)欄通知功能
這篇文章主要為大家詳細(xì)介紹了如何使用 wxPython 模塊,在 Windows 11 中實(shí)現(xiàn)任務(wù)欄通知功能,文中的示例代碼簡(jiǎn)潔易懂,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-10-10
python實(shí)現(xiàn)將excel文件轉(zhuǎn)化成CSV格式
下面小編就為大家分享一篇python實(shí)現(xiàn)將excel文件轉(zhuǎn)化成CSV格式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
Python3實(shí)現(xiàn)發(fā)送QQ郵件功能(html)
這篇文章主要為大家詳細(xì)介紹了Python3實(shí)現(xiàn)發(fā)送QQ郵件功能,html格式的qq郵件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12
python如何對(duì)數(shù)組進(jìn)行降維
這篇文章主要介紹了python如何對(duì)數(shù)組進(jìn)行降維問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05
Python神經(jīng)網(wǎng)絡(luò)TensorFlow基于CNN卷積識(shí)別手寫(xiě)數(shù)字
這篇文章主要介紹了Python神經(jīng)網(wǎng)絡(luò)TensorFlow基于CNN卷積識(shí)別手寫(xiě)數(shù)字的實(shí)現(xiàn)示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-10-10
python實(shí)現(xiàn)將range()函數(shù)生成的數(shù)字存儲(chǔ)在一個(gè)列表中
這篇文章主要介紹了python實(shí)現(xiàn)將range()函數(shù)生成的數(shù)字存儲(chǔ)在一個(gè)列表中,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-04-04

