欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python和Ruby中each循環(huán)引用變量問題(一個隱秘BUG?)

 更新時間:2014年06月04日 09:18:39   作者:  
這篇文章主要介紹了Python和Ruby中each循環(huán)引用變量問題,類似PHP的foreach中使用引用變量的問題,需要的朋友可以參考下

雖然這個問題我是在 Python 里遇到的,但是用 Ruby 解釋起來比較容易一些。在 Ruby 里,遍歷一個數(shù)組可以有很多種方法,最常用的兩種無非是 for 和 each:

復(fù)制代碼 代碼如下:

arr = ['a', 'b', 'c']

arr.each { |e|
  puts e
}

for e in arr
  puts e
end

通常我比較喜歡后者,似乎因為寫起來比較好看,不過從效率上來說前者應(yīng)該會稍微快一點,因為后者實際上是在遍歷的過程中對每個元素都調(diào)用一個 lambda 函數(shù)來做的,雖然一般情況下并不明顯,不過設(shè)置上下文并調(diào)用函數(shù)確實是有開銷的,特別是在動態(tài)語言里面(不考慮 JIT 內(nèi)聯(lián)優(yōu)化的話)。不過這次的問題并不是性能。然而確實跟“ each 對每個元素都會新建一個 scope 而 for 則不是”有關(guān)。

看下面一段代碼:

復(fù)制代碼 代碼如下:

arr = ['a', 'b', 'c']
h1 = Hash.new
h2 = Hash.new

arr.each { |e|
  h1[e] = lambda { e+'!'}
}

for e in arr
  h2[e] = lambda { e+'!' }
end

h1['a'].call # => ?
h2['a'].call # => ?

兩個 call 分別會得到什么?應(yīng)該已經(jīng)猜到了吧?分別是 'a!' 和 'c!' ,后者之所以是 'c!' 是因為 for 并沒有在循環(huán)的每一步都重新創(chuàng)建一個 scope ,因此三個 lambda 的 closure 引用到了同一個變量,而這個變量在最后一次被賦值為 'c' ,所以導(dǎo)致了這樣的后果。

問題其實出自我在用 Python 寫的一個小程序中的一段,代碼類似于這樣:

復(fù)制代碼 代碼如下:

for prop in public_props:
    setattr(proxy, 'get_%s'%prop, lambda: self.get_prop(prop))

其中 proxy 是我提供的一個代理對象,將 self 的一些公開的屬性給暴露出去,因為要限制對非 public 的屬性的訪問,我并不想在這個 proxy 中存放任何到 self 的引用,否則在沒有訪問權(quán)限限制的 Python 里通過類似 proxy._orig_self.some_private_prop 的方式來訪問是輕而易舉的。所以最后選擇了上面那樣的做法。

不幸的是,由于像剛才所說的那樣,for 并沒有每次都單獨創(chuàng)建 scope ,因此 closure 全部引用到了同一個變量上,導(dǎo)致所有的屬性值取出來都是最后一個屬性了??吹竭@樣詭異的 bug ,如果是在 C/C++ 里面,我肯定要懷疑是內(nèi)存或者指針的問題了。不過想了半天才終于恍然大悟!不過 Python 里面沒有 Ruby 那么方便的 each 可以用,lambda 用起來也很雞肋,所以最后通過定義一個局部的函數(shù)來解決了:

復(fù)制代碼 代碼如下:

def proxy_prop(name):
    setattr(proxy, 'get_%s'%prop, lambda: self.get_prop(name)
for prop in public_props:
    proxy_prop(prop)

最后,還要多嘴一句,對于之前 Ruby 那個例子,如果把 each 和 for 的執(zhí)行順序顛倒過來,會得到不同的結(jié)果:

復(fù)制代碼 代碼如下:
arr = ['a', 'b', 'c']
h1 = Hash.new
h2 = Hash.new

for e in arr
  h2[e] = lambda { e+'!' }
end

arr.each { |e|
  h1[e] = lambda { e+'!'}
}

h1['a'].call # => 'c!'
h2['a'].call # => 'c!'

現(xiàn)在兩個都是 'c!' 了!這是因為 Ruby 1.8 的實現(xiàn)里面 block 的參數(shù)可以對局部變量或者全局變量之類的任何東西進行賦值,而不是通常意義上的一個 lambda 函數(shù)的參數(shù)那么簡單。由于前面的 for 語句在當(dāng)前作用域創(chuàng)建了一個 e 作為局部變量,因此 each 就直接對這個局部變量進行賦值了,這樣,每次引用到的又變成了同一個東西,導(dǎo)致了一個隱秘的 Bug !

值得慶幸的是,block 的這個“特性”在 Ruby 1.9 中已經(jīng)被去除了,block 的參數(shù)只能是正常參數(shù),所以就不再存在這樣的問題了。希望 1.9 盡快普及吧!

相關(guān)文章

  • django安裝xadmin及問題解決

    django安裝xadmin及問題解決

    本文主要介紹了django安裝xadmin及問題解決,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • 詳解python--模擬輪盤抽獎游戲

    詳解python--模擬輪盤抽獎游戲

    這篇文章主要介紹了python模擬輪盤抽獎游戲,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • 使用Anaconda創(chuàng)建Pytorch虛擬環(huán)境的排坑詳細教程

    使用Anaconda創(chuàng)建Pytorch虛擬環(huán)境的排坑詳細教程

    PyTorch是一個開源的Python機器學(xué)習(xí)庫,基于Torch,用于自然語言處理等應(yīng)用程序,下面這篇文章主要給大家介紹了關(guān)于使用Anaconda創(chuàng)建Pytorch虛擬環(huán)境的相關(guān)資料,需要的朋友可以參考下
    2022-12-12
  • Python實現(xiàn)備份MySQL數(shù)據(jù)庫的方法示例

    Python實現(xiàn)備份MySQL數(shù)據(jù)庫的方法示例

    這篇文章主要介紹了Python實現(xiàn)備份MySQL數(shù)據(jù)庫的方法,涉及Python針對mysql數(shù)據(jù)庫的連接及基于mysqldump命令操作數(shù)據(jù)庫備份的相關(guān)實現(xiàn)技巧,需要的朋友可以參考下
    2018-01-01
  • Python高級特性與幾種函數(shù)的講解

    Python高級特性與幾種函數(shù)的講解

    今天小編就為大家分享一篇關(guān)于Python高級特性與幾種函數(shù)的講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-03-03
  • 解決python2 繪圖title,xlabel,ylabel出現(xiàn)中文亂碼的問題

    解決python2 繪圖title,xlabel,ylabel出現(xiàn)中文亂碼的問題

    今天小編就為大家分享一篇解決python2 繪圖title,xlabel,ylabel出現(xiàn)中文亂碼的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-01-01
  • pyinstaller封裝exe的操作

    pyinstaller封裝exe的操作

    這篇文章主要介紹了pyinstaller封裝exe的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-03-03
  • Python中的迭代器你了解嗎

    Python中的迭代器你了解嗎

    迭代器是一種特殊的對象,它實現(xiàn)了迭代協(xié)議,允許按照一定的順序逐個訪問元素,本文就來帶大家深入了解一下Python中迭代器的使用,需要的可以參考下
    2023-05-05
  • 在 Windows 下搭建高效的 django 開發(fā)環(huán)境的詳細教程

    在 Windows 下搭建高效的 django 開發(fā)環(huán)境的詳細教程

    這篇文章主要介紹了如何在 Windows 下搭建高效的 django 開發(fā)環(huán)境,本文通過一篇詳細教程實例代碼相結(jié)合給大家講解的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-07-07
  • python按照多個字符對字符串進行分割的方法

    python按照多個字符對字符串進行分割的方法

    這篇文章主要介紹了python按照多個字符對字符串進行分割的方法,涉及Python中正則表達式匹配的技巧,非常具有實用價值,需要的朋友可以參考下
    2015-03-03

最新評論