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

colorful

zc qq:1337220912

 

Lua closure upvalue (轉)

function是和數值、字符串同等地位的基本類型!

  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引用都會指向同一個地方。
  
  Lua將函數做為基本類型值并支持詞法定界的特性使得語言具有強大的抽象能力。而透徹認識函數、閉包和upvalue將幫助程序員善用這種能力。

posted on 2012-02-28 20:13 多彩人生 閱讀(2030) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


導航

統計

常用鏈接

留言簿(3)

隨筆分類

隨筆檔案

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲欧美在线观看| 欧美四级伦理在线| 亚洲国产婷婷| 久久久91精品| 久久精品一区二区三区不卡牛牛| 中文一区字幕| 欧美在线视频导航| 欧美aⅴ99久久黑人专区| 欧美激情一区二区三区在线视频观看| 欧美激情第1页| 一本色道婷婷久久欧美| 午夜日韩在线| 免费国产自线拍一欧美视频| 国产情人节一区| 精久久久久久久久久久| 亚洲精品在线一区二区| 亚洲一区二区在线播放| 国产欧美日韩视频一区二区三区| 国产性做久久久久久| 亚洲激情亚洲| 亚洲一区欧美激情| 美女精品在线| 亚洲资源在线观看| 欧美成人精品三级在线观看 | 亚洲一区二区三区在线播放| 久久精品欧美日韩精品| 亚洲精品一区二区在线| 久久国产66| 欧美天堂亚洲电影院在线播放| 国内精品伊人久久久久av一坑| 亚洲九九精品| 老鸭窝91久久精品色噜噜导演| 9色国产精品| 欧美大片在线看| 欧美在线观看网址综合| 一区二区三区在线视频免费观看| 国产一区二区三区四区在线观看| 亚洲国产专区校园欧美| 欧美中文字幕久久| 亚洲巨乳在线| 久久综合九色综合久99| 国产亚洲精品bv在线观看| 亚洲无吗在线| 亚洲精品黄色| 欧美成人精精品一区二区频| 一区福利视频| 久久精品日产第一区二区三区| 99精品欧美一区二区三区| 欧美高清在线一区| 亚洲国产成人在线播放| 亚洲第一二三四五区| 久久精彩视频| 亚洲欧美视频在线观看视频| 国产精品久久夜| 亚洲午夜激情免费视频| 99精品视频一区| 欧美三区美女| 亚洲综合成人婷婷小说| 在线视频亚洲一区| 国产精品久久二区二区| 亚洲专区欧美专区| 宅男噜噜噜66一区二区| 国产精品久久午夜夜伦鲁鲁| 香蕉成人久久| 亚洲欧美一区二区在线观看| 国产精品免费一区二区三区观看| 国产精品99久久久久久宅男| 一区二区欧美激情| 国产伦精品一区二区三区免费| 欧美在线国产| 久久se精品一区精品二区| 精品动漫3d一区二区三区免费| 免费日韩av电影| 欧美男人的天堂| 欧美亚洲综合网| 久久久综合视频| 亚洲美女免费精品视频在线观看| 亚洲精品免费在线| 国产精品久久久久久一区二区三区| 午夜精品理论片| 久久久97精品| 夜夜夜精品看看| 欧美一二三区精品| 亚洲国产精品精华液2区45| 亚洲欧洲精品一区二区三区波多野1战4 | 亚洲韩国精品一区| 亚洲日本免费| 国产精品亚洲成人| 亚洲一级在线观看| 欧美揉bbbbb揉bbbbb| 国产精品免费电影| 国产自产在线视频一区| 免费日本视频一区| 欧美日韩成人综合| 欧美专区亚洲专区| 欧美+日本+国产+在线a∨观看| 亚洲图中文字幕| 久久久久久久一区二区| 亚洲一区3d动漫同人无遮挡| 久久精品一区二区国产| 欧美精品国产一区二区| 久久精品最新地址| 欧美日本在线视频| 久热精品视频在线观看| 欧美三级第一页| 亚洲裸体在线观看| 久久久久久高潮国产精品视| 亚洲一区二区三区四区五区黄| 久久精品99久久香蕉国产色戒| 亚洲午夜未删减在线观看| 美女视频网站黄色亚洲| 久久精品人人做人人爽| 国产精品免费观看在线| 亚洲精品国产系列| 久久亚洲综合色| 欧美一级电影久久| 欧美午夜一区| 亚洲精品美女免费| 91久久精品国产91久久性色| 欧美在线视频a| 欧美一区二区性| 欧美性事在线| 亚洲精品一区二区在线观看| 亚洲精品国久久99热| 久久久久久有精品国产| 久久国产一区| 亚洲一区在线看| 亚洲一级黄色| 欧美日韩国产成人在线免费| 亚洲国产日韩欧美| 亚洲欧洲一区二区三区| 乱码第一页成人| 亚洲高清电影| 最新69国产成人精品视频免费| 久久在线播放| 欧美韩国在线| 欧美黑人国产人伦爽爽爽| 亚洲国产精品成人| 日韩视频免费观看高清在线视频| 欧美不卡在线| 亚洲高清不卡一区| 好看的亚洲午夜视频在线| 午夜精品国产| 久久亚洲美女| 91久久精品一区二区三区| 免费国产自线拍一欧美视频| 亚洲国产一二三| 亚洲毛片在线观看| 欧美精品粉嫩高潮一区二区| 亚洲免费观看视频| 亚洲欧美日本日韩| 国产一区二区毛片| 麻豆久久精品| 一区二区三区视频在线播放| 欧美在线视频a| 99国产精品国产精品久久| 亚洲欧美日韩成人| 久久精品国产v日韩v亚洲 | 免费在线一区二区| 亚洲精品少妇30p| 欧美一级大片在线免费观看| 国内外成人免费激情在线视频| 麻豆精品91| 亚洲素人一区二区| 女主播福利一区| 在线视频日韩精品| 国产亚洲精品久久久| 蜜臀av在线播放一区二区三区| 亚洲每日在线| 久久综合久久综合久久综合| 亚洲免费成人av| 国产情人综合久久777777| 免费精品99久久国产综合精品| 一区二区三区欧美成人| 免费不卡亚洲欧美| 亚洲免费中文| 日韩视频―中文字幕| 狠狠色伊人亚洲综合网站色| 欧美日韩综合不卡| 老牛影视一区二区三区| 亚洲男人的天堂在线aⅴ视频| 老**午夜毛片一区二区三区| 亚洲一区二区av电影| 亚洲国产91| 国产日韩在线不卡| 国产精品爱久久久久久久| 久久亚洲春色中文字幕久久久| 中文日韩在线| 亚洲欧洲在线观看| 欧美激情亚洲一区| 久久婷婷av| 欧美在线视频一区二区三区| 亚洲视频专区在线| 亚洲另类在线视频| 国产在线拍揄自揄视频不卡99 | 欧美日韩免费看| 欧美中日韩免费视频| 亚洲一区二区三区涩| 艳女tv在线观看国产一区| 亚洲国产欧美另类丝袜|