日韩久久久精品,亚洲精品久久久久久久久久久,亚洲欧美一区二区三区国产精品 ,一区二区福利

淺談使用Python變量時(shí)要避免的3個(gè)錯(cuò)誤

系統(tǒng) 1675 0

Python編程中經(jīng)常遇到一些莫名其妙的錯(cuò)誤, 其實(shí)這不是語(yǔ)言本身的問(wèn)題, 而是我們忽略了語(yǔ)言本身的一些特性導(dǎo)致的,今天就來(lái)看下使用Python變量時(shí)導(dǎo)致的3個(gè)不可思議的錯(cuò)誤, 以后在編程中要多多注意。

關(guān)于Python編程運(yùn)行時(shí)新手易犯錯(cuò)誤,這里暫不作介紹,詳情參見(jiàn):Python運(yùn)行的17個(gè)時(shí)新手常見(jiàn)錯(cuò)誤小結(jié)

1、 可變數(shù)據(jù)類型作為函數(shù)定義中的默認(rèn)參數(shù)

這似乎是對(duì)的?你寫(xiě)了一個(gè)小函數(shù),比如,搜索當(dāng)前頁(yè)面上的鏈接,并可選將其附加到另一個(gè)提供的列表中。

            
def search_for_links(page, add_to=[]):
  new_links = page.search_for_links()
  add_to.extend(new_links)
  return add_to
          

從表面看,這像是十分正常的 Python 代碼,事實(shí)上它也是,而且是可以運(yùn)行的。但是,這里有個(gè)問(wèn)題。如果我們給 add_to 參數(shù)提供了一個(gè)列表,它將按照我們預(yù)期的那樣工作。但是,如果我們讓它使用默認(rèn)值,就會(huì)出現(xiàn)一些神奇的事情。

試試下面的代碼:

            
def fn(var1, var2=[]):
  var2.append(var1)
  print(var2)
fn(3)
fn(4)
fn(5)
          

可能你認(rèn)為我們將看到:

            
[3]
[4]
[5]
          

但實(shí)際上,我們看到的卻是:

            
[3]
[3,4]
[3,4,5]
          

為什么呢?如你所見(jiàn),每次都使用的是同一個(gè)列表,輸出為什么會(huì)是這樣?在 Python 中,當(dāng)我們編寫(xiě)這樣的函數(shù)時(shí),這個(gè)列表被實(shí)例化為函數(shù)定義的一部分。當(dāng)函數(shù)運(yùn)行時(shí),它并不是每次都被實(shí)例化。這意味著,這個(gè)函數(shù)會(huì)一直使用完全一樣的列表對(duì)象,除非我們提供一個(gè)新的對(duì)象:

            
fn(3,[4])
[4,3]
          

答案正如我們所想的那樣。要想得到這種結(jié)果,正確的方法是:

            
def fn(var1, var2=None):
  ifnot var2:
    var2 =[]
  var2.append(var1)
          

或是在第一個(gè)例子中:

            
def search_for_links(page, add_to=None):
  ifnot add_to:
    add_to =[]
  new_links = page.search_for_links()
  add_to.extend(new_links)
  return add_to
          

這將在模塊加載的時(shí)候移走實(shí)例化的內(nèi)容,以便每次運(yùn)行函數(shù)時(shí)都會(huì)發(fā)生列表實(shí)例化。請(qǐng)注意,對(duì)于不可變數(shù)據(jù)類型,比如元組、字符串、整型,是不需要考慮這種情況的。這意味著,像下面這樣的代碼是非常可行的:

            
def func(message="my message"):
  print(message)
          

2、 可變數(shù)據(jù)類型作為類變量

這和上面提到的最后一個(gè)錯(cuò)誤很相像。思考以下代碼:

            
class URLCatcher(object):
  urls =[]
  def add_url(self, url):
    self.urls.append(url)
          

這段代碼看起來(lái)非常正常。我們有一個(gè)儲(chǔ)存 URL 的對(duì)象。當(dāng)我們調(diào)用 add_url 方法時(shí),它會(huì)添加一個(gè)給定的 URL 到存儲(chǔ)中。看起來(lái)非常正確吧?讓我們看看實(shí)際是怎樣的:

            
a =URLCatcher()
a.add_url('http://www.google.com')
b =URLCatcher()
b.add_url('http://www.pythontab.com')
print(b.urls)
print(a.urls)
          

結(jié)果:

            
['http://www.google.com','http://www.pythontab.com']
['http://www.google.com','http://www.pythontab.com']
          

等等,怎么回事?!我們想的不是這樣啊。我們實(shí)例化了兩個(gè)單獨(dú)的對(duì)象 a 和 b。把一個(gè) URL 給了 a,另一個(gè)給了 b。這兩個(gè)對(duì)象怎么會(huì)都有這兩個(gè) URL 呢?

這和第一個(gè)錯(cuò)例是同樣的問(wèn)題。創(chuàng)建類定義時(shí),URL 列表將被實(shí)例化。該類所有的實(shí)例使用相同的列表。在有些時(shí)候這種情況是有用的,但大多數(shù)時(shí)候你并不想這樣做。你希望每個(gè)對(duì)象有一個(gè)單獨(dú)的儲(chǔ)存。為此,我們修改代碼為:

            
class URLCatcher(object):
  def __init__(self):
    self.urls =[]
  def add_url(self, url):
    self.urls.append(url)
          

現(xiàn)在,當(dāng)創(chuàng)建對(duì)象時(shí),URL 列表被實(shí)例化。當(dāng)我們實(shí)例化兩個(gè)單獨(dú)的對(duì)象時(shí),它們將分別使用兩個(gè)單獨(dú)的列表。

3、 可變的分配錯(cuò)誤

這個(gè)問(wèn)題困擾了我一段時(shí)間。讓我們做出一些改變,并使用另一種可變數(shù)據(jù)類型 - 字典。

            
a ={'1':"one",'2':'two'}
          

現(xiàn)在,假設(shè)我們想把這個(gè)字典用在別的地方,且保持它的初始數(shù)據(jù)完整。

            
b = a
b['3']='three'
          

簡(jiǎn)單吧?

現(xiàn)在,讓我們看看原來(lái)那個(gè)我們不想改變的字典 a:

            
{'1':"one",'2':'two','3':'three'}
          

哇等一下,我們?cè)倏纯?b?

            
{'1':"one",'2':'two','3':'three'}
          

等等,什么?有點(diǎn)亂……讓我們回想一下,看看其它不可變類型在這種情況下會(huì)發(fā)生什么,例如一個(gè)元組:

            
c =(2,3)
d = c
d =(4,5)
          

現(xiàn)在 c 是 (2, 3),而 d 是 (4, 5)。

這個(gè)函數(shù)結(jié)果如我們所料。那么,在之前的例子中到底發(fā)生了什么?當(dāng)使用可變類型時(shí),其行為有點(diǎn)像 C 語(yǔ)言的一個(gè)指針。在上面的代碼中,我們令 b = a,我們真正表達(dá)的意思是:b 成為 a 的一個(gè)引用。它們都指向 Python 內(nèi)存中的同一個(gè)對(duì)象。聽(tīng)起來(lái)有些熟悉?那是因?yàn)檫@個(gè)問(wèn)題與先前的相似。

列表也會(huì)發(fā)生同樣的事嗎?是的。那么我們?nèi)绾谓鉀Q呢?這必須非常小心。如果我們真的需要復(fù)制一個(gè)列表進(jìn)行處理,我們可以這樣做:

            
b = a[:]
          

這將遍歷并復(fù)制列表中的每個(gè)對(duì)象的引用,并且把它放在一個(gè)新的列表中。但是要注意:如果列表中的每個(gè)對(duì)象都是可變的,我們將再次獲得它們的引用,而不是完整的副本。

假設(shè)在一張紙上列清單。在原來(lái)的例子中相當(dāng)于,A 某和 B 某正在看著同一張紙。如果有個(gè)人修改了這個(gè)清單,兩個(gè)人都將看到相同的變化。當(dāng)我們復(fù)制引用時(shí),每個(gè)人現(xiàn)在有了他們自己的清單。但是,我們假設(shè)這個(gè)清單包括尋找食物的地方。如果“冰箱”是列表中的第一個(gè),即使它被復(fù)制,兩個(gè)列表中的條目也都指向同一個(gè)冰箱。所以,如果冰箱被 A 修改,吃掉了里面的大蛋糕,B 也將看到這個(gè)蛋糕的消失。這里沒(méi)有簡(jiǎn)單的方法解決它。只要你記住它,并編寫(xiě)代碼的時(shí)候,使用不會(huì)造成這個(gè)問(wèn)題的方式。

字典以相同的方式工作,并且你可以通過(guò)以下方式創(chuàng)建一個(gè)昂貴副本:

            
b = a.copy()
          

再次說(shuō)明,這只會(huì)創(chuàng)建一個(gè)新的字典,指向原來(lái)存在的相同的條目。因此,如果我們有兩個(gè)相同的列表,并且我們修改字典 a 的一個(gè)鍵指向的可變對(duì)象,那么在字典 b 中也將看到這些變化。

可變數(shù)據(jù)類型的麻煩也是它們強(qiáng)大的地方。以上都不是實(shí)際中的問(wèn)題;它們是一些要注意防止出現(xiàn)的問(wèn)題。在第三個(gè)項(xiàng)目中使用昂貴復(fù)制操作作為解決方案在 99% 的時(shí)候是沒(méi)有必要的。

總結(jié)

以上就是本文關(guān)于淺談使用Python變量時(shí)要避免的3個(gè)錯(cuò)誤的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:python探索之BaseHTTPServer-實(shí)現(xiàn)Web服務(wù)器介紹、Python探索之SocketServer詳解等,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號(hào)聯(lián)系: 360901061

您的支持是博主寫(xiě)作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對(duì)您有幫助就好】

您的支持是博主寫(xiě)作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長(zhǎng)會(huì)非常 感謝您的哦!!!

發(fā)表我的評(píng)論
最新評(píng)論 總共0條評(píng)論
主站蜘蛛池模板: 黄大仙区| 界首市| 文昌市| 老河口市| 衡水市| 阜南县| 集安市| 呼伦贝尔市| 竹北市| 普安县| 井陉县| 名山县| 禹城市| 天气| 温泉县| 应城市| 库车县| 洞口县| 班玛县| 池州市| 日土县| 阿拉善盟| 丹寨县| 桓仁| 栖霞市| 什邡市| 宁河县| 南开区| 舒城县| 赤峰市| 黄山市| 池州市| 尚志市| 牙克石市| 盐山县| 遵义市| 晋州市| 崇信县| 佛冈县| 烟台市| 加查县|