寫在之前
在昨天的文章里 (零基礎(chǔ)學(xué)習(xí) Python 之字典)异雁,寫字典的方法的時候留了一個小尾巴瑰艘,那就是 copy() 方法還沒講是鬼。一是因為 copy 這個方法比較特殊,不單單是它表面的意思紫新;二是以為昨天的文章寫得比較長均蜜,可能你看到那的時候就沒啥耐心去仔細(xì)思考了,但是這個知識點又比較重要芒率,也是面試過程中會被長問起的題囤耳,我之前在面試的時候(干貨滿滿--親身經(jīng)歷的 Python 面試題)就被問起過。所以我把 copy 單獨摘出來今天單講敲董。
正式開始
首先我在這介紹兩個新的小知識紫皇,要在下面用到。一個是函數(shù) id() 腋寨,另一個是運算符 is聪铺。id() 函數(shù)就是返回對象的內(nèi)存地址;is 是比較兩個變量的對象引用是否指向同一個對象萄窜,在這里請不要和 == 混了铃剔,== 是比較兩個變量的值是否相等撒桨。
>>> a = [1,2,3]
>>> b = [1,2,3]
>>> id(a)
38884552L
>>> a is b
False
>>> a == b
True
</pre>
copy 這個詞有兩種叫法,一種是根據(jù)它的發(fā)音音譯過來的键兜,叫拷貝凤类;另一種就是標(biāo)準(zhǔn)的翻譯,叫復(fù)制普气。
其實單從表面意思來說谜疤,copy 就是將某件東西再復(fù)制一份,但是在很多編程語言中现诀,比如 Python夷磕,C++中,它就不是那么的簡單了仔沿。
>>> a = 1
>>> b = a
>>> b
1
看到上面的例子坐桩,從表面上看我們似乎是得到了兩個 1,但是如果你看過我之前寫的文章封锉,你應(yīng)該對一句話有印象绵跷,那就是 “變量無類型”, Python 中變量就是一個標(biāo)簽成福,這里我們有請 id() 閃亮登場碾局,看看它們在內(nèi)存中的位置。
>>> a = 1
>>> b = a
>>> b
1
>>> id(a)
31096808L
>>> id(b)
31096808L
看出來了嗎闷叉,id(a) 和 id(b) 相等擦俐,所以并沒有兩個 1,只是一個 1 而已握侧,只不過是在 1 上貼了兩張標(biāo)簽蚯瞧,名字是 a 和 b 罷了,這種現(xiàn)象普遍存在于 Python 之中品擎,這種賦值的方式實現(xiàn)了 “假裝” 拷貝埋合,真實的情況還是兩個變量和同一個對象之間的引用關(guān)系。
我們再來看 copy() 方法:
>>> a = {'name':'rocky','like':'python'}
>>> b = a.copy()
>>> b
{'name': 'rocky', 'like': 'python'}
>>> id(a)
31036280L
>>> id(b)
38786728L
咦萄传,果然這次得到的 b 和原來的 a 不同甚颂,它是在內(nèi)存中又開辟了一個空間。那么我們這個時候就來推理了秀菱,雖然它們兩個是一樣的振诬,但是它們在兩個不同的內(nèi)存空間里,那么肯定彼此互不干擾衍菱,如果我們?nèi)グ?b 改了赶么,那么 a 肯定不變。
>>> b['name'] = 'leey'
>>> b
{'name': 'leey', 'like': 'python'}
>>> a
{'name': 'rocky', 'like': 'python'}
結(jié)果和我們上面推理的一模一樣脊串,所以理解了對象有類型辫呻,變量無類型清钥,變量是對象的標(biāo)簽,就能正確推斷出 Python 提供的結(jié)果放闺。
我們接下來在看一個例子祟昭,請你在往下看的時候保證上面的你已經(jīng)懂了,不然容易暈車怖侦。
>>> a = {'name':'rocky','like':'python'}
>>> b = a
>>> b
{'name': 'rocky', 'like': 'python'}
>>> b['name'] = 'leey'
>>> b
{'name': 'leey', 'like': 'python'}
>>> a
{'name': 'leey', 'like': 'python'}
上面的例子看出什么來了嗎篡悟?修改了 b 對應(yīng)的字典類型的對象,a 的對象也變了础钠。也就是說恰力, b = a 得到的結(jié)果是兩個變量引用了同一個對象叉谜,但是事情真的這么簡單嗎旗吁?請睜大你的眼睛往下看,重點來了停局。
>>> first = {'name':'rocky','lanaguage':['python','c++','java']}
>>> second = first.copy()
>>> second
{'name': 'rocky', 'lanaguage': ['python', 'c++', 'java']}
>>> id(first)
31036280L
>>> id(second)
38786728L
在這里的話沒有問題很钓,和我們之前說的一樣,second 是從 first 拷貝過來的董栽,它們分別引用的是兩個對象码倦。
>>> second['lanaguage'].remove('java')
>>> second
{'name': 'rocky', 'lanaguage': ['python', 'c++']}
>>> first
{'name': 'rocky', 'lanaguage': ['python', 'c++']}
發(fā)現(xiàn)什么了嗎?按理說上述例子中 second 的 lanaguage 對應(yīng)的是一個列表锭碳,我刪除這個列表里的值袁稽,也只應(yīng)該改變的是 second 啊,為什么連 first 的也會改擒抛,不是應(yīng)該互不干擾嗎推汽?是不是很意外?是我們之前說的不對嗎歧沪?那我們再試試另一個鍵:
>>> second['name'] = 'leey'
>>> second
{'name': 'leey', 'lanaguage': ['python', 'c++']}
>>> first
{'name': 'rocky', 'lanaguage': ['python', 'c++']}
前面說的原理是有效的歹撒,那這到底是為什么啊,來來來诊胞,有請我們的 id() 再次閃亮登場暖夭。
>>> id(first['name'])
38829152L
>>> id(second['name'])
38817544L
>>> id(first['lanaguage'])
38754120L
>>> id(second['lanaguage'])
38754120L
其實這里深層次的原因是和 Python 的存儲數(shù)據(jù)的方式有關(guān),這里不做過多的說明(其實是我也不懂撵孤。迈着。 在這里,我們只需要知道的是邪码,當(dāng) copy() 的時候裕菠,列表這類由字符串,數(shù)字等復(fù)合而成的對象仍然是復(fù)制了引用霞扬,也就是貼標(biāo)簽糕韧,并沒有建立一個新的對象枫振,我們把這種拷貝方式叫做淺拷貝(唉呀媽呀,終于把這個概念引出來了萤彩。粪滤。,言外之意就是并沒有解決深層次的問題雀扶,再言外之意就是還有能夠解決深層次問題的方法杖小。
確實,在 Python 中還有一個深拷貝(deep copy)愚墓,在使用它之前要引入一個 copy 模塊予权,我們來試一下。
>>> import copy
>>> first = {'name':'rocky','lanaguage':['python','c++','java']}
>>> second = copy.deepcopy(first)
>>> second
{'name': 'rocky', 'lanaguage': ['python', 'c++', 'java']}
>>> second['lanaguage'].remove('java')
>>> second
{'name': 'rocky', 'lanaguage': ['python', 'c++']}
>>> first
{'name': 'rocky', 'lanaguage': ['python', 'c++', 'java']}
用了深拷貝以后浪册,果然就不是引用了扫腺。
寫在最后
深拷貝和淺拷貝到這里就講完了,花了一番功夫總算寫的還令自己滿意村象,不知道朋友們看到這里的時候是否是覺得對這一部分豁然開朗笆环,我盡力了。這個拓展也可能是成為一個系列厚者,補(bǔ)充一些我覺得理解起來比較困難或者平時面試求職或者工作中常見的知識點躁劣,希望您多捧場。
最后感謝你能看到這里库菲,希望我寫的東西能夠讓你有到收獲账忘,但是我還是希望我在文章里插入的代碼,你們能自己動手試一下熙宇,都很簡單鳖擒。原創(chuàng)不易,每一個字奇颠,每一個標(biāo)點都是自己手敲的败去,所以希望大家能多給點支持,該關(guān)注關(guān)注烈拒,該點贊點贊圆裕,該轉(zhuǎn)發(fā)轉(zhuǎn)發(fā),有什么問題歡迎在后臺聯(lián)系我荆几,也可以在公眾號 -- Python空間 找到我的微信加我吓妆。
The end。