在分享每個(gè)Python新手應(yīng)該知道的4個(gè)常見(jiàn)錯(cuò)誤之前,請(qǐng)確保您熟悉以下文章中的一些Python內(nèi)置功能。
1.不使用迭代器
每個(gè)Python新手都會(huì)這樣做,無(wú)論他們是否熟練使用其他編程語(yǔ)言。 跑不了的。
給定一個(gè)列表list_,您將如何使用for循環(huán)逐個(gè)訪問(wèn)列表中的元素? 我們知道Python中的列表已建立索引,因此我們可以通過(guò)list_ [i]訪問(wèn)第i個(gè)元素。 然后,我們可以為for循環(huán)創(chuàng)建一個(gè)介于0到len(list_)之間的整數(shù)的迭代器,如下所示:
for i in range(len(list_)): foo(list_[i])
有用。 代碼沒(méi)有問(wèn)題。 這也是在其他語(yǔ)言(例如C)中構(gòu)造for循環(huán)的標(biāo)準(zhǔn)方法。但是實(shí)際上,我們可以在Python中做得更好。
怎么樣?
您知道Python中的列表是可迭代的嗎? 通過(guò)利用其可迭代的性質(zhì),我們可以生成更具可讀性的代碼,如下所示:
for element in list_: foo(element)
Photo by The Creative Exchange on Unsplash
通過(guò)zip函數(shù)可以在for循環(huán)中并行遍歷多個(gè)列表,而如果您堅(jiān)持在迭代可迭代對(duì)象時(shí)獲取索引號(hào)(即計(jì)數(shù)器),則枚舉可能會(huì)有所幫助。 我希望早先了解的5個(gè)Python功能對(duì)它們進(jìn)行了介紹和解釋。
2.使用全局變量
全局變量是在主腳本中聲明的具有全局范圍的變量,而局部變量是在具有局部范圍的函數(shù)內(nèi)聲明的變量。 在Python中使用global關(guān)鍵字可讓您在函數(shù)中本地訪問(wèn)和更改全局變量。 這是一個(gè)例子:
a = 1 # a variable def increment(): a += 1 return adef increment2(): global a # can make changes to global variable “a” a += 1 return a increment() # UnboundLocalError: local variable ‘a(chǎn)’ referenced before assignmentincrement2() # returns 2
許多初學(xué)者都喜歡它,因?yàn)槭褂胓lobal似乎可以避免傳遞函數(shù)所需的所有參數(shù)。 但這實(shí)際上是不正確的。 它只是隱藏了動(dòng)作。
使用全局變量也不利于調(diào)試。 功能應(yīng)被視為功能塊框,并且應(yīng)可重復(fù)使用。 修改全局變量的函數(shù)可能會(huì)給很難發(fā)現(xiàn)的主腳本帶來(lái)副作用,并且可能導(dǎo)致復(fù)雜的意大利面條式代碼,并且調(diào)試起來(lái)要困難得多。
在局部函數(shù)中修改全局變量是不良的編程習(xí)慣。 您應(yīng)該將變量作為參數(shù)傳遞,對(duì)其進(jìn)行修改,并在函數(shù)末尾將其返回。
Photo by Vladislav Klapin on Unsplash
*不要將全局變量與全局常量混淆,因?yàn)樵诖蠖鄶?shù)情況下使用后者非常好。
3.不了解可變對(duì)象
對(duì)于新的Python學(xué)習(xí)者來(lái)說(shuō),這也許是最常見(jiàn)的驚喜,因?yàn)榇斯δ茉谠撜Z(yǔ)言中非常獨(dú)特。
Python中有兩種對(duì)象。 可變對(duì)象可以在運(yùn)行時(shí)更改其狀態(tài)或內(nèi)容,而不可變對(duì)象則不能。 許多內(nèi)置對(duì)象類型是不可變的,包括int,float,string,bool和tuple。
st = ‘A string’ st[0] = ‘B’ # You cannot do this in Python
另一方面,諸如list,set和dict的數(shù)據(jù)類型是可變的。 因此,您可以更改列表中元素的內(nèi)容,例如 list_ [0] =‘new’。
如果函數(shù)中的默認(rèn)參數(shù)是可變的,則會(huì)發(fā)生意外情況。 讓我們以以下函數(shù)為例,其中可變的空列表是參數(shù)list_的默認(rèn)值。
def foo(element, list_=[]): list_.append(element) r eturn list_
讓我們兩次調(diào)用該函數(shù),而不用輸入list_的參數(shù),以使其采用默認(rèn)值。 理想情況下,如果不提供第二個(gè)參數(shù),則每次調(diào)用該函數(shù)時(shí)都會(huì)創(chuàng)建一個(gè)新的空列表。
a = foo(1) # returns [1]b = foo(2) # returns [1,2], not [2]! WHY?
什么?
事實(shí)證明,在定義函數(shù)時(shí),Python中的默認(rèn)參數(shù)會(huì)被評(píng)估一次。 這意味著調(diào)用該函數(shù)不會(huì)刷新其默認(rèn)參數(shù)。
Photo by Ravi Roshan on Unsplash
因此,如果默認(rèn)參數(shù)是可變的,并且每次調(diào)用該函數(shù)時(shí)都會(huì)將其更改。可變的默認(rèn)參數(shù)將適用于所有將來(lái)的函數(shù)調(diào)用。 “標(biāo)準(zhǔn)”解決方案是使用(不可變)None默認(rèn)值,如下所示。
def foo(element, list_=None): if list_ is None: list_ = [] list_.append(element) return list_
4.不復(fù)制
復(fù)制的概念對(duì)于學(xué)習(xí)者而言可能是陌生的,甚至是違反直覺(jué)的。 假設(shè)您有一個(gè)列表a = [[0,1],[2,3]],然后通過(guò)b = a聲明一個(gè)新列表。 現(xiàn)在,您將擁有兩個(gè)具有相同元素的列表。 通過(guò)更改列表b中的某些元素,它應(yīng)該不會(huì)對(duì)列表a產(chǎn)生任何(副作用),對(duì)嗎?
錯(cuò)誤。
a = [[0,1],[2,3]]b = ab[1][1] = 100print(a,b) # [[0, 1], [2, 100]] [[0, 1], [2, 100]]print(id(a)==id(b))# True
當(dāng)您使用賦值語(yǔ)句(即b = a)“復(fù)制”列表時(shí),在一個(gè)列表元素上所做的任何修改在兩個(gè)列表中均可見(jiàn)。 賦值運(yùn)算符僅在目標(biāo)和對(duì)象之間創(chuàng)建綁定,因此示例中的列表a和b共享相同的引用,即Python中的id()。
如何復(fù)制對(duì)象?
如果您要“復(fù)制”對(duì)象并且僅修改新(或舊)對(duì)象中的值而沒(méi)有綁定,則有兩種創(chuàng)建副本的方法:淺副本和深副本。 兩個(gè)對(duì)象將具有不同的引用。
Photo by Louis Hansel on Unsplash
使用前面的示例,可以通過(guò)b = copy.copy(a)創(chuàng)建a的淺表副本。 淺表副本會(huì)創(chuàng)建一個(gè)新對(duì)象,該對(duì)象存儲(chǔ)原始元素的引用。 這聽(tīng)起來(lái)可能很復(fù)雜,但讓我們看下面的示例:
import copya = [[0,1],[2,3]]b = copy.copy(a)print(id(a)==id(b))# Falseb[1] = 100print(a,b)# [[0, 1], [2, 3]] [[0, 1], 100]b[0][0] = -999print(a,b)# [[-999, 1], [2, 3]] [[-999, 1], 100]print(id(a[0]) == id(b[0]))# True
在創(chuàng)建嵌套列表a的淺副本(我們稱為b)之后,兩個(gè)列表具有不同的引用id(a)!= id(b),符號(hào)!=表示“不等于”。 但是,它們的元素具有相同的引用,因此id(a [0])== id(b [0])。
這意味著更改b內(nèi)部的元素不會(huì)影響列表a,但是修改b [1]內(nèi)部的元素確實(shí)會(huì)影響a [1],因此此副本很淺。
簡(jiǎn)而言之,如果b是a的淺副本,則對(duì)b中的嵌套對(duì)象內(nèi)的元素進(jìn)行的任何更改都將顯示在a中。
如果要復(fù)制嵌套對(duì)象而元素之間沒(méi)有任何綁定,則需要使用b = copy.deepcopy(a)的深拷貝。 深層副本將創(chuàng)建一個(gè)新對(duì)象,然后以遞歸方式在原始元素中創(chuàng)建嵌套對(duì)象的副本。
簡(jiǎn)而言之,深拷貝復(fù)制所有內(nèi)容而沒(méi)有任何綁定。
-
函數(shù)
+關(guān)注
關(guān)注
3文章
4343瀏覽量
62809 -
代碼
+關(guān)注
關(guān)注
30文章
4808瀏覽量
68816 -
python
+關(guān)注
關(guān)注
56文章
4801瀏覽量
84865
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論