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

Python延遲綁定問題原理及解決方案

 更新時間:2020年08月04日 11:57:27   投稿:yaominghui  
這篇文章主要介紹了Python延遲綁定問題原理及解決方案,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下

延遲綁定出現(xiàn)在閉包問題中。下面我們看一個閉包的例子:

def (n):
  def mul(x):
    return n*x
  return mul
double = gen_mul(2)
doubled_value = double(6) 

可以看出滿足閉包的幾點:

  • 有內(nèi)部函數(shù)
  • 內(nèi)部函數(shù)引用了外部函數(shù)中的自由變量
  • 內(nèi)部函數(shù)被返回

閉包的優(yōu)點:

  • 可以避免使用全局變量
  • 可以持久化變量,達到靜態(tài)變量的作用

閉包的缺點:

  • 可能會消耗大量的內(nèi)存
  • 可能會導(dǎo)致內(nèi)存泄漏

當然缺點可以通過人為避免。

現(xiàn)在我們來看看另一個會引出延遲綁定的例子:

def multipliers():
  return [lambda x : i * x for i in range(4)]
print([m(2) for m in multipliers()]) # [6,6,6,6]

上邊的例子會輸出[6,6,6,6],而不是我們預(yù)期的[0,2,4,6]。

這就是延遲綁定導(dǎo)致的結(jié)果。具體過程我們可以來分析下:
執(zhí)行第三行時,會先執(zhí)行multipliers函數(shù),然后執(zhí)行函數(shù)中的列表解析式。在每一次迭代的時候都會生成一個匿名函數(shù)(這里只是定義)作為元素。然后回到第三行,遍歷返回的列表中的匿名函數(shù),傳入?yún)?shù)2并執(zhí)行。此時函數(shù)類似于這樣:

def noname(x):
return i * x

我們知道Python查找變量的作用域鏈的順序依次為LEGB:

局部變量(L)->外部函數(shù)中的局部變量(E)->全局變量(G)->內(nèi)置變量(B)

非常重要的一點我們需要知道:Python的作用域在編譯時就已經(jīng)形成了,而不是在運行時,函數(shù)的作用域與其被調(diào)用的位置無關(guān)。

那么在本例中,上面的noname函數(shù)體中的i從何而來呢?當然首先會到multipliers函數(shù)的局部變量中去尋找。此時i的值已經(jīng)為3,所以出現(xiàn)這種讓人”費解”的現(xiàn)象。

那么現(xiàn)在我們既然已經(jīng)知道了原因,那么要怎樣解決呢?

我們可以將迭代的i值直接注入到匿名函數(shù)的函數(shù)體中,這里給出兩種方法:

通過為參數(shù)設(shè)置默認值,這是因為在編譯時就會計算確定默認值:

def multipliers_ch1():
return [lambda m,x=i : m * x for i in range(4)]

通過內(nèi)置函數(shù)partial:

from functools import partial
def multipliers_ch2():
  return [partial(lambda m,x : m * x,i) for i in range(4)]

利用生成器的延遲計算:

def multipliers_ch3():
  for m in range(4):
    yield lambda x: m * x

partial及生成器的內(nèi)容會在以后分享。

運行結(jié)果

print([m(2) for m in multipliers_ch1()]) # [0,2,4,6]
print([m(2) for m in multipliers_ch2()]) # [0,2,4,6]
print([m(2) for m in multipliers_ch3()]) # [0,2,4,6]

注:

自由變量:指未在本地作用域中綁定的變量,我們可通過訪問函數(shù)的code屬性進行查看:

fun.code.co_freevars

LEGB: 可看該部分解釋

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • 利用Pandas讀取文件路徑或文件名稱包含中文的csv文件方法

    利用Pandas讀取文件路徑或文件名稱包含中文的csv文件方法

    今天小編就為大家分享一篇利用Pandas讀取文件路徑或文件名稱包含中文的csv文件方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-07-07
  • pandas添加新列的5種常見方法

    pandas添加新列的5種常見方法

    pandas為DataFrame格式數(shù)據(jù)添加新列的方法非常簡單,下面這篇文章主要給大家介紹了關(guān)于pandas添加新列的5種常見方法,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-06-06
  • python concurrent.futures模塊的使用測試

    python concurrent.futures模塊的使用測試

    大家都知道concurrent.futures 是 3.2 中引入的新模塊,它為異步執(zhí)行可調(diào)用對象提供了高層接口,今天通過本文給大家介紹python concurrent.futures模塊的使用測試 ,感興趣的朋友一起看看吧
    2021-07-07
  • django配置連接數(shù)據(jù)庫及原生sql語句的使用方法

    django配置連接數(shù)據(jù)庫及原生sql語句的使用方法

    這篇文章主要給大家介紹了關(guān)于django配置連接數(shù)據(jù)庫,以及原生sql語句的使用方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • 使用Python繪制圖表大全總結(jié)

    使用Python繪制圖表大全總結(jié)

    本篇文章主要介紹了使用Python繪制圖表大全總結(jié),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-02-02
  • python中reload(module)的用法示例詳解

    python中reload(module)的用法示例詳解

    與from和import相比,reload是內(nèi)置函數(shù),而不是語句,下面這篇文章主要給大家介紹了關(guān)于python中reload(module)用法的相關(guān)資料,文中給出了詳細的示例代碼供大家參考學(xué)習(xí),需要的朋友們下面來一起看看吧。
    2017-09-09
  • django 配置阿里云OSS存儲media文件的例子

    django 配置阿里云OSS存儲media文件的例子

    今天小編就為大家分享一篇django 配置阿里云OSS存儲media文件的例子,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-08-08
  • Pandas DataFrame操作數(shù)據(jù)增刪查改

    Pandas DataFrame操作數(shù)據(jù)增刪查改

    我們在用 pandas 處理數(shù)據(jù)的時候,經(jīng)常會遇到用其中一列數(shù)據(jù)替換另一列數(shù)據(jù)的場景。這一類的需求估計很多人都遇到,當然還有其它更復(fù)雜的。解決這類需求的辦法有很多,這里我們來推薦幾個,這篇文章主要介紹了Pandas DataFrame操作數(shù)據(jù)的增刪查改
    2022-10-10
  • python操作XML格式文件的一些常見方法

    python操作XML格式文件的一些常見方法

    最近有同學(xué)詢問如何利用Python處理xml文件,特此整理一篇比較簡潔的操作手冊,下面這篇文章主要給大家介紹了關(guān)于python操作XML格式文件的一些常見方法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-07-07
  • 如何基于Python深度圖生成3D點云詳解

    如何基于Python深度圖生成3D點云詳解

    通常使用TOF等3d攝像頭采集的格式一般只是深度圖,下面這篇文章主要給大家介紹了關(guān)于如何基于Python深度圖生成3D點云的相關(guān)資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-12-12

最新評論