
深入探究Python中變量的拷貝和作用域問題
在 python 中賦值語句總是建立對象的引用值,而不是復(fù)制對象。因此,python 變量更像是指針,而不是數(shù)據(jù)存儲區(qū)域,
這點(diǎn)和大多數(shù) OO 語言類似吧,比如 C++、java 等 ~
1、先來看個(gè)問題吧:
在Python中,令values=[0,1,2];values[1]=values,為何結(jié)果是[0,[...],2]?
>>> values = [0, 1, 2]
>>> values[1] = values
>>> values
[0, [...], 2]
我預(yù)想應(yīng)當(dāng)是
[0, [0, 1, 2], 2]
但結(jié)果卻為何要賦值無限次?
可以說 Python 沒有賦值,只有引用。你這樣相當(dāng)于創(chuàng)建了一個(gè)引用自身的結(jié)構(gòu),所以導(dǎo)致了無限循環(huán)。為了理解這個(gè)問題,有個(gè)基本概念需要搞清楚。
Python 沒有「變量」,我們平時(shí)所說的變量其實(shí)只是「標(biāo)簽」,是引用。
執(zhí)行
values = [0, 1, 2]
的時(shí)候,Python 做的事情是首先創(chuàng)建一個(gè)列表對象 [0, 1, 2],然后給它貼上名為 values 的標(biāo)簽。如果隨后又執(zhí)行
values = [3, 4, 5]
的話,Python 做的事情是創(chuàng)建另一個(gè)列表對象 [3, 4, 5],然后把剛才那張名為 values 的標(biāo)簽從前面的 [0, 1, 2] 對象上撕下來,重新貼到 [3, 4, 5] 這個(gè)對象上。
至始至終,并沒有一個(gè)叫做 values 的列表對象容器存在,Python 也沒有把任何對象的值復(fù)制進(jìn) values 去。過程如圖所示:
執(zhí)行
的時(shí)候,Python 做的事情則是把 values 這個(gè)標(biāo)簽所引用的列表對象的第二個(gè)元素指向 values 所引用的列表對象本身。執(zhí)行完畢后,values 標(biāo)簽還是指向原來那個(gè)對象,只不過那個(gè)對象的結(jié)構(gòu)發(fā)生了變化,從之前的列表 [0, 1, 2] 變成了 [0, ?, 2],而這個(gè) ? 則是指向那個(gè)對象本身的一個(gè)引用。如圖所示:
要達(dá)到你所需要的效果,即得到 [0, [0, 1, 2], 2] 這個(gè)對象,你不能直接將 values[1] 指向 values 引用的對象本身,而是需要吧 [0, 1, 2] 這個(gè)對象「復(fù)制」一遍,得到一個(gè)新對象,再將 values[1] 指向這個(gè)復(fù)制后的對象。Python 里面復(fù)制對象的操作因?qū)ο箢愋投悾瑥?fù)制列表 values 的操作是
values[:] #生成對象的拷貝或者是復(fù)制序列,不再是引用和共享變量,但此法只能頂層復(fù)制
所以你需要執(zhí)行
Python 做的事情是,先 dereference 得到 values 所指向的對象 [0, 1, 2],然后執(zhí)行 [0, 1, 2][:] 復(fù)制操作得到一個(gè)新的對象,內(nèi)容也是 [0, 1, 2],然后將 values 所指向的列表對象的第二個(gè)元素指向這個(gè)復(fù)制二來的列表對象,最終 values 指向的對象是 [0, [0, 1, 2], 2]。過程如圖所示:
往更深處說,values[:] 復(fù)制操作是所謂的「淺復(fù)制」(shallow copy),當(dāng)列表對象有嵌套的時(shí)候也會產(chǎn)生出乎意料的錯(cuò)誤,比如
問:此時(shí) a 和 b 分別是多少?
正確答案是 a 為 [8, [1, 9], 3],b 為 [0, [1, 9], 3]。發(fā)現(xiàn)沒?b 的第二個(gè)元素也被改變了。想想是為什么?不明白的話看下圖
正確的復(fù)制嵌套元素的方法是進(jìn)行「深復(fù)制」(deep copy),方法是
2、引用 VS 拷貝:
(1)沒有限制條件的分片表達(dá)式(L[:])能夠復(fù)制序列,但此法只能淺層復(fù)制。
(2)字典 copy 方法,D.copy() 能夠復(fù)制字典,但此法只能淺層復(fù)制
(3)有些內(nèi)置函數(shù),例如 list,能夠生成拷貝 list(L)
(4)copy 標(biāo)準(zhǔn)庫模塊能夠生成完整拷貝:deepcopy 本質(zhì)上是遞歸 copy
(5)對于不可變對象和可變對象來說,淺復(fù)制都是復(fù)制的引用,只是因?yàn)閺?fù)制不變對象和復(fù)制不變對象的引用是等效的(因?yàn)閷ο蟛豢勺?,?dāng)改變時(shí)會新建對象重新賦值)。所以看起來淺復(fù)制只復(fù)制不可變對象(整數(shù),實(shí)數(shù),字符串等),對于可變對象,淺復(fù)制其實(shí)是創(chuàng)建了一個(gè)對于該對象的引用,也就是說只是給同一個(gè)對象貼上了另一個(gè)標(biāo)簽而已。
L = [1, 2, 3]
D = {'a':1, 'b':2}
A = L[:]
B = D.copy()
print "L, D"
print L, D
print "A, B"
print A, B
print "--------------------"
A[1] = 'NI'
B['c'] = 'spam'
print "L, D"
print L, D
print "A, B"
print A, B
L, D
[1, 2, 3] {'a': 1, 'b': 2}
A, B
[1, 2, 3] {'a': 1, 'b': 2}
--------------------
L, D
[1, 2, 3] {'a': 1, 'b': 2}
A, B
[1, 'NI', 3] {'a': 1, 'c': 'spam', 'b': 2}
3、增強(qiáng)賦值以及共享引用:
x = x + y,x 出現(xiàn)兩次,必須執(zhí)行兩次,性能不好,合并必須新建對象 x,然后復(fù)制兩個(gè)列表合并
屬于復(fù)制/拷貝
x += y,x 只出現(xiàn)一次,也只會計(jì)算一次,性能好,不生成新對象,只在內(nèi)存塊末尾增加元素。
當(dāng) x、y 為list時(shí), += 會自動調(diào)用 extend 方法進(jìn)行合并運(yùn)算,in-place change。
屬于共享引用
L = [1, 2]
M = L
L = L + [3, 4]
print L, M
print "-------------------"
L = [1, 2]
M = L
L += [3, 4]
print L, M
[1, 2, 3, 4] [1, 2]
-------------------
[1, 2, 3, 4] [1, 2, 3, 4]
4、python 從2.x 到3.x,語句變函數(shù)引發(fā)的變量作用域問題
先看段代碼:
def test():
a = False
exec ("a = True")
print ("a = ", a)
test()
b = False
exec ("b = True")
print ("b = ", b)
在 python 2.x 和 3.x 下 你會發(fā)現(xiàn)他們的結(jié)果不一樣:
2.x:
a = True
b = True
3.x:
a = False
b = True
這是為什么呢?
因?yàn)?3.x 中 exec 由語句變成函數(shù)了,而在函數(shù)中變量默認(rèn)都是局部的,也就是說
你所見到的兩個(gè) a,是兩個(gè)不同的變量,分別處于不同的命名空間中,而不會沖突。
具體參考 《learning python》P331-P332
知道原因了,我們可以這么改改:
def test():
a = False
ldict = locals()
exec("a=True",globals(),ldict)
a = ldict['a']
print(a)
test()
b = False
exec("b = True", globals())
print("b = ", b)
這個(gè)問題在 stackoverflow 上已經(jīng)有人問了,而且 python 官方也有人報(bào)了 bug。。。
數(shù)據(jù)分析咨詢請掃描二維碼
若不方便掃碼,搜微信號:CDAshujufenxi
LSTM 模型輸入長度選擇技巧:提升序列建模效能的關(guān)鍵? 在循環(huán)神經(jīng)網(wǎng)絡(luò)(RNN)家族中,長短期記憶網(wǎng)絡(luò)(LSTM)憑借其解決長序列 ...
2025-07-11CDA 數(shù)據(jù)分析師報(bào)考條件詳解與準(zhǔn)備指南? ? 在數(shù)據(jù)驅(qū)動決策的時(shí)代浪潮下,CDA 數(shù)據(jù)分析師認(rèn)證愈發(fā)受到矚目,成為眾多有志投身數(shù) ...
2025-07-11數(shù)據(jù)透視表中兩列相乘合計(jì)的實(shí)用指南? 在數(shù)據(jù)分析的日常工作中,數(shù)據(jù)透視表憑借其強(qiáng)大的數(shù)據(jù)匯總和分析功能,成為了 Excel 用戶 ...
2025-07-11尊敬的考生: 您好! 我們誠摯通知您,CDA Level I和 Level II考試大綱將于 2025年7月25日 實(shí)施重大更新。 此次更新旨在確保認(rèn) ...
2025-07-10BI 大數(shù)據(jù)分析師:連接數(shù)據(jù)與業(yè)務(wù)的價(jià)值轉(zhuǎn)化者? ? 在大數(shù)據(jù)與商業(yè)智能(Business Intelligence,簡稱 BI)深度融合的時(shí)代,BI ...
2025-07-10SQL 在預(yù)測分析中的應(yīng)用:從數(shù)據(jù)查詢到趨勢預(yù)判? ? 在數(shù)據(jù)驅(qū)動決策的時(shí)代,預(yù)測分析作為挖掘數(shù)據(jù)潛在價(jià)值的核心手段,正被廣泛 ...
2025-07-10數(shù)據(jù)查詢結(jié)束后:分析師的收尾工作與價(jià)值深化? ? 在數(shù)據(jù)分析的全流程中,“query end”(查詢結(jié)束)并非工作的終點(diǎn),而是將數(shù) ...
2025-07-10CDA 數(shù)據(jù)分析師考試:從報(bào)考到取證的全攻略? 在數(shù)字經(jīng)濟(jì)蓬勃發(fā)展的今天,數(shù)據(jù)分析師已成為各行業(yè)爭搶的核心人才,而 CDA(Certi ...
2025-07-09【CDA干貨】單樣本趨勢性檢驗(yàn):捕捉數(shù)據(jù)背后的時(shí)間軌跡? 在數(shù)據(jù)分析的版圖中,單樣本趨勢性檢驗(yàn)如同一位耐心的偵探,專注于從單 ...
2025-07-09year_month數(shù)據(jù)類型:時(shí)間維度的精準(zhǔn)切片? ? 在數(shù)據(jù)的世界里,時(shí)間是最不可或缺的維度之一,而year_month數(shù)據(jù)類型就像一把精準(zhǔn) ...
2025-07-09CDA 備考干貨:Python 在數(shù)據(jù)分析中的核心應(yīng)用與實(shí)戰(zhàn)技巧? ? 在 CDA 數(shù)據(jù)分析師認(rèn)證考試中,Python 作為數(shù)據(jù)處理與分析的核心 ...
2025-07-08SPSS 中的 Mann-Kendall 檢驗(yàn):數(shù)據(jù)趨勢與突變分析的有力工具? ? ? 在數(shù)據(jù)分析的廣袤領(lǐng)域中,準(zhǔn)確捕捉數(shù)據(jù)的趨勢變化以及識別 ...
2025-07-08備戰(zhàn) CDA 數(shù)據(jù)分析師考試:需要多久?如何規(guī)劃? CDA(Certified Data Analyst)數(shù)據(jù)分析師認(rèn)證作為國內(nèi)權(quán)威的數(shù)據(jù)分析能力認(rèn)證 ...
2025-07-08LSTM 輸出不確定的成因、影響與應(yīng)對策略? 長短期記憶網(wǎng)絡(luò)(LSTM)作為循環(huán)神經(jīng)網(wǎng)絡(luò)(RNN)的一種變體,憑借獨(dú)特的門控機(jī)制,在 ...
2025-07-07統(tǒng)計(jì)學(xué)方法在市場調(diào)研數(shù)據(jù)中的深度應(yīng)用? 市場調(diào)研是企業(yè)洞察市場動態(tài)、了解消費(fèi)者需求的重要途徑,而統(tǒng)計(jì)學(xué)方法則是市場調(diào)研數(shù) ...
2025-07-07CDA數(shù)據(jù)分析師證書考試全攻略? 在數(shù)字化浪潮席卷全球的當(dāng)下,數(shù)據(jù)已成為企業(yè)決策、行業(yè)發(fā)展的核心驅(qū)動力,數(shù)據(jù)分析師也因此成為 ...
2025-07-07剖析 CDA 數(shù)據(jù)分析師考試題型:解鎖高效備考與答題策略? CDA(Certified Data Analyst)數(shù)據(jù)分析師考試作為衡量數(shù)據(jù)專業(yè)能力的 ...
2025-07-04SQL Server 字符串截取轉(zhuǎn)日期:解鎖數(shù)據(jù)處理的關(guān)鍵技能? 在數(shù)據(jù)處理與分析工作中,數(shù)據(jù)格式的規(guī)范性是保證后續(xù)分析準(zhǔn)確性的基礎(chǔ) ...
2025-07-04CDA 數(shù)據(jù)分析師視角:從數(shù)據(jù)迷霧中探尋商業(yè)真相? 在數(shù)字化浪潮席卷全球的今天,數(shù)據(jù)已成為企業(yè)決策的核心驅(qū)動力,CDA(Certifie ...
2025-07-04CDA 數(shù)據(jù)分析師:開啟數(shù)據(jù)職業(yè)發(fā)展新征程? ? 在數(shù)據(jù)成為核心生產(chǎn)要素的今天,數(shù)據(jù)分析師的職業(yè)價(jià)值愈發(fā)凸顯。CDA(Certified D ...
2025-07-03