青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Lua中的函數是一階類型值(first-class value),定義函數就象創建普通類型值一樣(只不過函數類型值的數據主要是一條條指令而已),所以在函數體中仍然可以定義函數。假設函數f2定義在函數f1中,那么就稱f2為f1的內嵌(inner)函數,f1為f2的外包(enclosing)函數,外包和內嵌都具有傳遞性,即f2的內嵌必然是f1的內嵌,而f1的外包也一定是f2的外包。內嵌函數可以訪問外包函數已經創建的所有局部變量,這種特性便是所謂的詞法定界(lexical scoping),而這些局部變量則稱為該內嵌函數的外部局部變量(external local variable)或者upvalue(這個詞多少會讓人產生誤解,因為upvalue實際指的是變量而不是值)。試看如下代碼:

function f1(n)
   -- 函數參數也是局部變量

   local function f2()
      print(n) -- 引用外包函數的局部變量
   end
   return f2
end

g1 = f1(1979)
g1() -- 打印出1979
g2 = f1(500)
g2() -- 打印出500


當執行完g1 = f1(1979)后,局部變量n的生命本該結束,但因為它已經成了內嵌函數f2(它又被賦給了變量g1)的upvalue,所以它仍然能以某種形式繼續“存活”下來,從而令g1()打印出正確的值。

    可為什么g2與g1的函數體一樣(都是f1的內嵌函數f2的函數體),但打印值不同?這就涉及到一個相當重要的概念——閉包(closure)。事實上,Lua編譯一個函數時,會為它生成一個原型(prototype),其中包含了函數體對應的虛擬機指令、函數用到的常量值(數,文本字符串等等)和一些調試信息。在運行時,每當Lua執行一個形如function...end 這樣的表達式時,它就會創建一個新的數據對象,其中包含了相應函數原型的引用、環境(environment,用來查找全局變量的表)的引用以及一個由所有upvalue引用組成的數組,而這個數據對象就稱為閉包。由此可見,函數是編譯期概念,是靜態的,而閉包是運行期概念,是動態的。g1和g2的值嚴格來說不是函數而是閉包,并且是兩個不相同的閉包,而每個閉包可以保有自己的upvalue值,所以g1和g2打印出的結果當然就不一樣了。雖然閉包和函數是本質不同的概念,但為了方便,且在不引起混淆的情況下,我們對它們不做區分。

    使用upvalue很方便,但它們的語義也很微妙,需要引起注意。比如將f1函數改成:

function f1(n)
   local function f2()
      print(n)
   end
   n = n + 10
   return f2
end

g1 = f1(1979)
g1() -- 打印出1989


內嵌函數定義在n = n + 10這條語句之前,可為什么g1()打印出的卻是1989?upvalue實際是局部變量,而局部變量是保存在函數堆棧框架上(stack frame)的,所以只要upvalue還沒有離開自己的作用域,它就一直生存在函數堆棧上。這種情況下,閉包將通過指向堆棧上的upvalue的引用來訪問它們,一旦upvalue即將離開自己的作用域(這也意味著它馬上要從堆棧中消失),閉包就會為它分配空間并保存當前的值,以后便可通過指向新分配空間的引用來訪問該upvalue。當執行到f1(1979)的n = n + 10時,閉包已經創建了,但是n并沒有離開作用域,所以閉包仍然引用堆棧上的n,當return f2完成時,n即將結束生命,此時閉包便將n(已經是1989了)復制到自己管理的空間中以便將來訪問。弄清楚了內部的秘密后,運行結果就不難解釋了。

upvalue還可以為閉包之間提供一種數據共享的機制。試看下例:

function Create(n)
   local function foo1()
      print(n)
   end

   local function foo2()
      n = n + 10
   end

   return foo1,foo2
end

f1,f2 = Create(1979)
f1() -- 打印1979
f2()
f1() -- 打印1989
f2()
f1() -- 打印1999


f1,f2這兩個閉包的原型分別是Create中的內嵌函數foo1和foo2,而foo1和foo2引用的upvalue是同一個,即Create的局部變量n。前面已說過,執行完Create調用后,閉包會把堆棧上n的值復制出來,那么是否f1和f2就分別擁有一個n的拷貝呢?其實不然,當Lua發現兩個閉包的upvalue指向的是當前堆棧上的相同變量時,會聰明地只生成一個拷貝,然后讓這兩個閉包共享該拷貝,這樣任一個閉包對該upvalue進行修改都會被另一個探知。上述例子很清楚地說明了這點:每次調用f2都將upvalue的值增加了10,隨后f1將更新后的值打印出來。upvalue的這種語義很有價值,它使得閉包之間可以不依賴全局變量進行通訊,從而使代碼的可靠性大大提高。

    閉包在創建之時其upvalue就已經不在堆棧上的情況也有可能發生,這是因為內嵌函數可以引用更外層外包函數的局部變量:

function Test(n)
   local function foo()
      local function inner1()
         print(n)
      end
      local function inner2()
         n = n + 10
      end
      return inner1,inner2
   end
   return foo
end

t = Test(1979)
f1,f2 = t()
f1()        -- 打印1979
f2()
f1()        -- 打印1989
g1,g2 = t()
g1()        -- 打印1989
g2()
g1()        -- 打印1999
f1()        -- 打印1999


執行完t = Test(1979)后,Test的局部變量n就“死”了,所以當f1,f2這兩個閉包被創建時堆棧上根本找不到n的蹤影,這叫它們如何取得n的值呢?呵呵,不要忘了Test函數的n不僅僅是inner1和inner2的upvalue,同時它也是foo的upvalue。t = Test(1979)之后,t這個閉包一定已經把n妥善保存好了,之后f1、f2如果在當前堆棧上找不到n就會自動到它們的外包閉包(姑且這么叫)的upvalue引用數組中去找,并把找到的引用值拷貝到自己的upvalue引用數組中。仔細觀察上述代碼,可以判定g1和g2與f1和f2共享同一個upvalue。這是為什么呢?其實,g1和g2與f1和f2都是同一個閉包(t)創建的,所以它們引用的upvalue(n)實際也是同一個變量,而剛才描述的搜索機制則保證了最后它們的upvalue引用都會指向同一個地方。

  

posted on 2007-04-02 23:21 清源游民 閱讀(2527) 評論(11)  編輯 收藏 引用 所屬分類: Lua

FeedBack:
# re: 學習lua中的閉包
2007-04-03 08:59 | 夢在天涯
我還不之lua是什么東東那,哈哈,有機會學習哦  回復  更多評論
  
# re: 學習lua中的閉包
2007-04-27 11:39 | melodie
釋我惑也,你是怎么分析出他的機制的?對于習慣了c/c++這種不可嵌套定義的語言的程序員來說,理解閉包真不太容易。我琢磨了半天,感覺它很類似于c++中的functor  回復  更多評論
  
# re: 學習lua中的閉包
2007-08-06 10:39 | 阿來
老大寫得文章真不錯!不頂下實在過意不去!  回復  更多評論
  
# re: 學習lua中的閉包
2007-09-23 22:49 | 絕對在乎你
這個東西比較有用,很多時候可以作為像C++的模板的方式使用。

最近一直在使用lua做應用,有空多交流!  回復  更多評論
  
# re: 學習lua中的閉包
2009-04-30 15:54 | zdy
不錯的文章,頂一下  回復  更多評論
  
# re: 學習lua中的閉包
2009-08-21 03:38 | XX
很好啊,謝謝。看了更明白了  回復  更多評論
  
# re: 學習lua中的閉包
2009-10-29 11:05 | 炮灰九段
f1(1979)() <-> 1989
f1(1979)() <-> 1989
閉包是妙,不知道應用如何
  回復  更多評論
  
# re: 學習lua中的閉包
2010-09-06 22:47 | tinggo
找了好久,終于找到能夠深刻解釋閉包內部機制的文章。就算是3年后仍然是一篇很好的文章,但是請問樓主,Lua資料較少,您是怎么獲得這些信息的?  回復  更多評論
  
# re: 學習lua中的閉包
2011-08-16 16:50 | xiaoboz
function f1(n)
local function f2()
print(n)
end
n = n + 10
return f2
end

g1 = f1(1979)
g1()--輸出1989
g1(2000)--輸出1989
  回復  更多評論
  
# re: 學習lua中的閉包
2011-10-07 02:50 |
頂,真的很不錯  回復  更多評論
  
# re: 學習lua中的閉包
2011-10-07 02:53 |
@xiaoboz
因為g1()并不需要接受參數,而你傳了一個參數,按照lua語法,你傳的參數等于作廢,所以不起作用
  回復  更多評論
  
<2009年10月>
27282930123
45678910
11121314151617
18192021222324
25262728293031
1234567

留言簿(35)

隨筆分類(78)

隨筆檔案(74)

文章檔案(5)

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲免费播放| 久久成人精品无人区| 亚洲国内自拍| 欧美专区在线播放| 99av国产精品欲麻豆| 亚洲欧洲一区二区在线观看| 久久九九免费| 中国女人久久久| 欧美阿v一级看视频| 国内精品久久久久影院色 | 国产精品爽爽爽| 日韩视频免费看| 欧美激情精品久久久久久大尺度| 久久久久国产精品厨房| 国产无遮挡一区二区三区毛片日本| 亚洲伊人久久综合| 欧美怡红院视频| 亚洲免费在线视频一区 二区| 国产精品高潮呻吟久久av黑人| 亚洲色在线视频| 亚洲精品一区二| 欧美日韩精品一区二区在线播放| 日韩午夜精品视频| 日韩午夜免费视频| 欧美日韩国产大片| 亚洲欧美国产毛片在线| 亚洲四色影视在线观看| 国产精品美女久久久| 西西裸体人体做爰大胆久久久| 亚洲视频在线一区| 久久精品91| 在线日本高清免费不卡| 欧美aaaaaaaa牛牛影院| 欧美高清你懂得| 一本一道久久综合狠狠老精东影业| 亚洲韩国一区二区三区| 欧美日韩视频一区二区| 香蕉成人啪国产精品视频综合网| 欧美影视一区| 亚洲日本欧美| 亚洲图片欧美一区| 狠狠色狠色综合曰曰| 午夜精品999| 久久黄色级2电影| 亚洲国产乱码最新视频| 一区二区三区日韩欧美| 国外成人在线视频网站| 亚洲国产欧美另类丝袜| 国产精品日日摸夜夜添夜夜av| 久久婷婷影院| 欧美日韩在线播放三区| 亚洲狠狠丁香婷婷综合久久久| 日韩视频在线一区二区三区| 国产一本一道久久香蕉| 亚洲激情一区二区| 国产偷国产偷亚洲高清97cao| 欧美成人激情视频免费观看| 欧美日韩国产在线| 久久躁日日躁aaaaxxxx| 欧美日韩国产综合新一区| 久久精品中文字幕一区| 米奇777在线欧美播放| 欧美一区二区视频免费观看| 美女精品视频一区| 亚洲国产成人在线| 久久天天躁狠狠躁夜夜爽蜜月| 国产精品va在线| 久久在线免费观看| 国产精品初高中精品久久| 久久一区二区视频| 欧美午夜片欧美片在线观看| 欧美成人综合一区| 国产伦精品一区二区三区免费| 夜夜狂射影院欧美极品| 久久亚洲国产精品日日av夜夜| 亚洲欧美在线x视频| 欧美电影免费观看高清| 久久久久免费视频| 国产精品视频免费在线观看| 亚洲国产日韩在线| 伊人久久综合97精品| 亚洲一区二区三区在线视频| 一区二区三区视频在线观看| 免费日韩一区二区| 久久香蕉精品| 欧美私人网站| 亚洲精品国偷自产在线99热| 在线看日韩av| 久久精品国产69国产精品亚洲| 欧美一区成人| 国产精品久久久久毛片软件| 一本色道久久综合亚洲精品婷婷 | 久久色在线播放| 噜噜噜在线观看免费视频日韩| 国产精品日韩欧美一区二区| 亚洲视频一区二区在线观看| 一区二区欧美亚洲| 欧美新色视频| 在线视频一区观看| 亚洲午夜精品久久| 欧美日一区二区三区在线观看国产免| 亚洲第一黄网| 亚洲伦理在线免费看| 欧美激情按摩在线| 亚洲人成网站999久久久综合| 亚洲国产精品成人va在线观看| 久久视频这里只有精品| 欧美高清在线视频观看不卡| 亚洲茄子视频| 欧美精品一区二区在线观看 | 久久人人爽人人| 免费在线亚洲欧美| 亚洲国产欧美一区二区三区久久 | 亚洲影院免费观看| 欧美影院成人| 樱花yy私人影院亚洲| 蜜桃精品久久久久久久免费影院| 欧美好骚综合网| 亚洲婷婷综合色高清在线| 欧美日韩中文在线观看| 先锋资源久久| 牛牛影视久久网| 夜夜爽av福利精品导航| 国产免费一区二区三区香蕉精| 亚洲专区一区| 99国产精品视频免费观看| 一级日韩一区在线观看| 国产精品网站在线| 久久精视频免费在线久久完整在线看| 老**午夜毛片一区二区三区| 日韩午夜高潮| 国产精品网站在线观看| 久久久久久亚洲精品杨幂换脸| 亚洲国产日韩美| 先锋a资源在线看亚洲| 亚洲欧洲日本国产| 欧美日韩亚洲综合一区| 欧美亚洲视频一区二区| 亚洲国产精品一区制服丝袜| 亚洲欧美视频在线观看| 亚洲电影免费在线| 国产精品一区二区三区四区五区| 久久蜜臀精品av| 亚洲图片欧洲图片av| 亚洲国产91精品在线观看| 亚洲在线观看视频网站| 91久久久国产精品| 国产一区二区三区成人欧美日韩在线观看 | 国产麻豆91精品| 免费一级欧美片在线观看| 亚洲一区二区不卡免费| 蜜臀va亚洲va欧美va天堂| 亚洲午夜成aⅴ人片| 国产私拍一区| 欧美wwwwww| 欧美综合国产| 99精品热6080yy久久 | 亚洲精品之草原avav久久| 久久嫩草精品久久久久| 一区在线播放| 国产精品一区亚洲| 欧美揉bbbbb揉bbbbb| 久久综合精品一区| 性久久久久久久| 欧美一区二区三区男人的天堂| 亚洲美女毛片| 精品动漫av| 红杏aⅴ成人免费视频| 老司机一区二区| 欧美专区日韩视频| 性欧美xxxx大乳国产app| 日韩视频―中文字幕| 91久久国产综合久久蜜月精品| 欧美在线网站| 欧美在线免费观看| 欧美在线资源| 一区二区免费在线观看| 一个色综合av| 亚洲激情视频在线播放| 永久免费精品影视网站| 欧美日韩一级片在线观看| 欧美a级一区| 欧美高潮视频| 蜜桃av综合| 欧美国产一区在线| 久久久久久香蕉网| 老司机午夜精品| 欧美成年人网站| 美女主播视频一区| 欧美日韩国产成人在线| 欧美成人亚洲| 欧美日韩综合不卡| 久久精品噜噜噜成人av农村| 欧美在线综合视频| 久久美女性网| 午夜精品福利电影| 亚洲视频碰碰| 国产精品99久久99久久久二8| 亚洲一区二区av电影| 一区二区三区久久久|