python for循環(huán)remove同一個(gè)list過程解析
下午在用python將Linux的conf配置文件轉(zhuǎn)化成字典dict時(shí)遇到了一個(gè)奇怪的問題,原先conf配置文件中沒有注釋行(以#開頭的行),后來為了避免這種情況,添加了一個(gè)對(duì)以#開頭的行刪除的操作。 實(shí)踐結(jié)果顛覆了已有的認(rèn)知,直接上代碼示例。
代碼片段1
#!/usr/bin/python # encoding: utf-8 # -*- coding: utf8 -*- import re list_to_test = ['# ', '# conf', 'NAME="Ubuntu"', 'VERSION="14.04.3 LTS, Trusty Tahr"'] for member in list_to_test: if re.search('^#+.*', member) is not None: list_to_test.remove(member) print list_to_test
結(jié)果1:
['# conf', 'NAME="Ubuntu"', 'VERSION="14.04.3 LTS, Trusty Tahr"']
代碼片段2
#!/usr/bin/python # encoding: utf-8 # -*- coding: utf8 -*- list_to_test = ['# ', '# conf', 'NAME="Ubuntu"', 'VERSION="14.04.3 LTS, Trusty Tahr"'] list_to_test.remove('# ') list_to_test.remove('# conf') print list_to_test
# 結(jié)果2:
['NAME="Ubuntu"', 'VERSION="14.04.3 LTS, Trusty Tahr"']
本以為上述兩個(gè)代碼的結(jié)果應(yīng)該是一樣的,結(jié)果不一樣。
分析:
原因是不能在for循環(huán)中用remove同一個(gè)列表(遍歷中刪除)。當(dāng)remove這個(gè)list中的元素時(shí),list的長度發(fā)生了變化,for循環(huán)就會(huì)受到影響(這個(gè)python版本(2.7.x沒有明顯的報(bào)錯(cuò),可能作者并不認(rèn)為這是一個(gè)issue或bug,但給點(diǎn)提示也是好的啊)。
解決辦法:
用一個(gè)新的列表(list)去代替循環(huán)中的list或者代替remove操作的list。在創(chuàng)建新的列表是可以用cpoy模塊中的deepcopy方法也可以用new_list = old_list[:]的方法,如下:
#!/usr/bin/python # encoding: utf-8 # -*- coding: utf8 -*- import re from copy import deepcopy old_list = ['# ', '# conf', 'NAME="Ubuntu"', 'VERSION="14.04.3 LTS, Trusty Tahr"'] new_list = deepcopy(old_list) for member in new_list: if re.search('^#+.*', member) is not None: old_list.remove(member) print old_list
有趣(令人困惑)的是切片也是淺復(fù)制,但利用切片也可以實(shí)現(xiàn)上述功能,代碼如下:
#!/usr/bin/python # encoding: utf-8 # -*- coding: utf8 -*- import re old_list = ['# ', '# conf', 'NAME="Ubuntu"', 'VERSION="14.04.3 LTS, Trusty Tahr"'] new_list = old_list[:] for member in new_list: if re.search('^#+.*', member) is not None: old_list.remove(member) print old_list
上述導(dǎo)致錯(cuò)誤發(fā)生的例子(在for循環(huán)中用remove同一個(gè)列表)可以認(rèn)知為這個(gè)操作是修改對(duì)象勢必影響此對(duì)象,要想修改一個(gè)對(duì)象卻不影響此對(duì)象引用,則需要對(duì)象復(fù)制。如果你想修改一個(gè)對(duì)象,而且想讓原始的對(duì)象不受影響,那你就需要對(duì)象復(fù)制。
附加知識(shí)點(diǎn):
關(guān)于淺復(fù)制(淺拷貝)
對(duì)象的淺復(fù)制(shallow copy):它雖然復(fù)制了對(duì)象,但對(duì)于對(duì)象中的元素,依然使用引用.
(1)、使用切片[:]操作進(jìn)行拷貝 (注釋:切片只復(fù)制了對(duì)象的頂層,對(duì)對(duì)象的下一層還是引用,舉個(gè)例子:[1,2,3,[4,5,6]])
(2)、使用工廠函數(shù)(如list/dir/set)等進(jìn)行拷貝
(3)、copy.copy()
(4)、=(賦值)操作(注釋:原文沒有,此處是新添加的,根據(jù)“對(duì)象的賦值實(shí)際上是對(duì)象的引用”添加)
如果希望復(fù)制一個(gè)容器對(duì)象,以及它里面的所有元素(包含元素的子元素),使用copy.deepcopy這個(gè)方法會(huì)消耗一些時(shí)間和空間。不過,如果你需要完全復(fù)制,這是唯一的方法。
注意:
1、對(duì)于非容器類型(如數(shù)字、字符串、和其他‘原子'類型的對(duì)象)沒有被拷貝一說。
2、如果元組變量只包含原子類型對(duì)象,則不能深copy。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Python List remove()實(shí)例用法詳解
- Python 刪除List元素的三種方法remove、pop、del
- Python如何在循環(huán)內(nèi)使用list.remove()
- Python列表list內(nèi)建函數(shù)用法實(shí)例分析【insert、remove、index、pop等】
- python list count統(tǒng)計(jì)個(gè)數(shù)的實(shí)現(xiàn)
- Python中List.count()方法的使用教程
- python學(xué)習(xí)-List移除某個(gè)值remove和統(tǒng)計(jì)值次數(shù)count
相關(guān)文章
python實(shí)現(xiàn)銀行管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)銀行管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-10-10深入解析Python設(shè)計(jì)模式編程中建造者模式的使用
這篇文章主要介紹了深入解析Python設(shè)計(jì)模式編程中建造者模式的使用,建造者模式的程序通常將所有細(xì)節(jié)都交由子類實(shí)現(xiàn),需要的朋友可以參考下2016-03-03tesseract庫及訓(xùn)練數(shù)據(jù)下載安裝方式
這篇文章主要介紹了tesseract庫及訓(xùn)練數(shù)據(jù)下載安裝方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02