偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Python為什么引入這兩個(gè)關(guān)鍵詞:global和nonlocal

開發(fā) 后端
本文通過分析函數(shù)對(duì)全局變量和閉包變量的讀、寫操作,借助于反匯編字節(jié)碼分析,認(rèn)清了 global 和 nonlocal 關(guān)鍵字的用法,對(duì)其引入和作用有了較為深刻認(rèn)識(shí)。

[[408478]]

本文轉(zhuǎn)載自微信公眾號(hào)「Python中文社區(qū)」,作者鞏慶奎。轉(zhuǎn)載本文請(qǐng)聯(lián)系Python中文社區(qū)公眾號(hào)。

啥是 global 和 nonlocal

Python 支持的關(guān)鍵詞里,global 和 nonlocal 初學(xué)者接觸的少,不知道是做什么用的;一些人雖然知道它們的作用,但對(duì)為什么要引入這兩個(gè)關(guān)鍵詞則有些不知其所以然。

粗淺地說,global 和 nonlocal 是為了在函數(shù)中修改全局和閉包變量而引入的關(guān)鍵字。

本文用代碼一點(diǎn)點(diǎn)分析引入 global 和 nonlocal 的原因。

一個(gè)奇怪的現(xiàn)象

下面,讓我們做一個(gè)測(cè)試。

  1. g =1 
  2. def fun(): 
  3.     g = 2 
  4.     return g  
  5. print(fun(),g) 

一般地,我們認(rèn)為結(jié)果應(yīng)該為 2,2。這一點(diǎn)學(xué)過其他語言如 Java、c 的同學(xué)尤其認(rèn)同。

但讓我們跑起來,可以看到結(jié)果為 2,1。也就是說,函數(shù)沒有改變?nèi)肿兞?g。

這很奇怪,究其原因是因?yàn)椋?/p>

  • Python 認(rèn)為所有 = 賦值都是在當(dāng)前作用域新建變量。
    • 當(dāng)我們?cè)诔绦蛑?g = 1 時(shí),表示當(dāng)前全局作用域建立 g,賦值 1。
    • 當(dāng)我們?cè)诤瘮?shù)中 g = 2 時(shí),表示當(dāng)前局部作用域建立 g,賦值 2。

使用 dis.dis(fun) 分析 fun 函數(shù)源代碼:

  1. 16           0 LOAD_CONST               1 (2) 
  2.               2 STORE_FAST               0 (g) 
  3.  17           4 LOAD_FAST                0 (g) 
  4.               6 RETURN_VALUE 

可見,第 2 條指令 STORE_FAST,這是存儲(chǔ)到局部變量的命令。

所以,函數(shù)中實(shí)際操作的是局部變量。

還有更甚的例子如下,大家猜測(cè)下執(zhí)行結(jié)果。

  1. g =1 
  2. def fun(): 
  3.     g += 1 
  4.     return g 
  5. print(fun(),g)  

根據(jù)上文,我們知道函數(shù)不會(huì)改變?nèi)肿兞?g,那么結(jié)果應(yīng)該是 2,1,這次總算對(duì)了吧?

很抱歉,當(dāng)執(zhí)行到 g += 1 時(shí),系統(tǒng)報(bào)錯(cuò):UnboundLocalError: local variable 'g' referenced before assignment。

仔細(xì)觀察錯(cuò)誤,local variable 'g',這里的 g 仍然被視為局部變量:沒有定義(=賦值),就直接 inplace add,當(dāng)然要報(bào)錯(cuò)。

也就是說,所有在局部作用域中對(duì)全局變量的賦值、原位賦值都會(huì)失敗。唯有如下函數(shù)給我們帶來一絲安慰。

  1. g =1 
  2. def fun(): 
  3.     return g 
  4. print(fun(),g)  

結(jié)果 1,1,總算還有個(gè)正常的:在局部作用域中引用全局作用域變量正常。

那當(dāng)我必須修改全局變量時(shí),該怎么辦呢?

global 的引入和分析

這就是 global 引入的理由了,將全局變量擴(kuò)展到函數(shù)中來,使函數(shù)可以修改全局變量。

  1. g =1 
  2. def fun(): 
  3.     global g 
  4.     g = 2 
  5.     return g 
  6. print(fun(),g)  

結(jié)果為 2,2,函數(shù)修改了全局變量。我們來看 dis.dis(fun) 的反匯編代碼。

  1. 37           0 LOAD_CONST               1 (2) 
  2.              2 STORE_GLOBAL             0 (g) 
  3. 38           4 LOAD_GLOBAL              0 (g) 
  4.              6 RETURN_VALUE 

第 2 條指令,STORE_GLOBAL 是將常量 2 賦值給全局變量 g,異于上例中的 STORE_FAST指令對(duì)局部變量操作。

故此,我們得出結(jié)論:當(dāng)在函數(shù)中讀取全局變量時(shí),可以直接使用。但如果需要修改全局變量值,則需要在變量前加上 global 來修飾。

nonlocal 的引入

同樣地,當(dāng)我們書寫嵌套函數(shù),需要對(duì)閉包中的變量進(jìn)行修改操作時(shí),我們也需要引入 nonlocal 關(guān)鍵字。

如下函數(shù)中,我們定義了閉包,閉包中的變量 e,試圖在內(nèi)嵌函數(shù)中進(jìn)行修改,但沒有使用 nonlocal 關(guān)鍵字聲明 e。

  1. def outer(): 
  2.     e = 1 
  3.     def inner(): 
  4.         e = 2 
  5.         return e 
  6.     return inner 

參照上例,我們知道這種修改是徒勞的——因?yàn)榭捶磪R編代碼 dis.dis(outer()) 可知:

  1. 63           0 LOAD_CONST               1 (2) 
  2.              2 STORE_FAST               0 (e) 
  3. 64           4 LOAD_FAST                0 (e) 
  4.              6 RETURN_VALUE 

第 2 條指令 STORE_FAST,操作局部變量,也就是說 inner 里的 e,仍然被視為局部變量。

雷同于上例的 global,這里使用 nonlocal 來在內(nèi)嵌函數(shù) inner 中修改閉包變量 e。

  1. def outer(): 
  2.     e = 1 
  3.     def inner(): 
  4.         nonlocal e 
  5.         e = 2 
  6.         return e 
  7.     return inner 

查看此時(shí)的反匯編代碼 dis.dis(outer()) 可知:

  1. 78           0 LOAD_CONST               1 (2) 
  2.              2 STORE_DEREF              0 (e) 
  3. 79           4 LOAD_DEREF               0 (e) 
  4.              6 RETURN_VALUE 

第 2 條指令 STORE_DEREF,操作的是閉包變量,也就是說 inner 里的 e,是可以修改的閉包中的 e。

總結(jié)

本文通過分析函數(shù)對(duì)全局變量和閉包變量的讀、寫操作,借助于反匯編字節(jié)碼分析,認(rèn)清了 global 和 nonlocal 關(guān)鍵字的用法,對(duì)其引入和作用有了較為深刻認(rèn)識(shí)。

 

作者:鞏慶奎,大奎,對(duì)計(jì)算機(jī)、電子信息工程感興趣。

 

責(zé)任編輯:武曉燕 來源: Python中文社區(qū)
相關(guān)推薦

2011-06-20 14:32:59

關(guān)鍵詞

2011-06-07 18:45:41

關(guān)鍵詞

2011-06-14 19:11:38

關(guān)鍵詞

2016-06-02 11:02:02

機(jī)器學(xué)習(xí)

2011-06-16 17:54:25

關(guān)鍵詞

2011-06-21 16:11:04

SEO關(guān)鍵詞

2011-05-25 17:58:00

2013-08-26 15:43:40

AppStore關(guān)鍵詞開發(fā)者應(yīng)用選取關(guān)鍵詞

2019-12-22 13:48:26

退休科技行業(yè)大佬

2011-05-25 17:38:56

關(guān)鍵詞

2023-01-13 16:57:50

SpringBoot配置核心

2020-12-10 10:32:33

區(qū)塊鏈比特幣數(shù)字貨幣

2016-01-06 10:05:20

2013-04-16 21:52:27

IBM云計(jì)算開源

2011-05-10 10:39:07

關(guān)鍵詞SEO

2011-06-19 12:20:47

長(zhǎng)尾關(guān)鍵詞

2012-03-20 22:19:16

Linux

2011-06-14 10:01:03

長(zhǎng)尾關(guān)鍵詞

2009-11-03 09:02:59

Windows 7蘋果競(jìng)爭(zhēng)

2011-07-22 15:48:46

SEO
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)