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

Lua中的協同程序探究

 更新時間:2014年09月13日 10:56:20   作者:笨木頭  
這篇文章主要介紹了Lua中的協同程序探究,本文講解了什么是協同程序、創(chuàng)建協同程序、更像樣的協同程序、讓協同程序掛起、resume操作的返回值,需要的朋友可以參考下

哎,周五晚上我都還這么努力看書,真是好孩子。(小若:不想吐槽了)

其實我都準備玩游戲看電影去的了,但是這書就擺在桌子上,而且正對著我,就想著,掃兩眼吧。

結果一掃就不對勁了,因為這內容有點繞,有點小混亂,如果我現在不記錄下來的話,下周一可能又要重新看一次了。
 
好吧,今天我們來聊聊協同程序。

1.什么是協同程序(coroutinue)

大家都知道線程吧?都知道多線程吧?協同程序就和這線程差不多,但是又有比較明顯的區(qū)別。

多個協同程序在任意時刻只能執(zhí)行一個,雖然線程在某種意義上也是這樣,但這不是一樣的概念。

換句話說,一個協同程序在運行的時候,其他協同程序是無法獲得執(zhí)行的機會的。
只有正在運行的協同程序主動掛起時,其他協同程序才有機會執(zhí)行。
 
而線程呢?即使不主動休眠,也很有可能因為輪片時間到達而把執(zhí)行機會讓給其他線程。

2.創(chuàng)建協同程序

創(chuàng)建協同程序很簡單,咋一看,其實和線程沒差別~
代碼如下:

復制代碼 代碼如下:

local co = coroutine.create(function() print("hello coroutine"); end);

協同的程序的操作都在coroutine里,create函數的參數就是協同程序要執(zhí)行的函數,就這么運行代碼是沒有效果的。

因為協同程序創(chuàng)建后,默認是掛起狀態(tài)。

協同程序的四種狀態(tài)分別為:掛起(suspended)、運行(running)、死亡(dead)、正常(normal)。
 
要想協同程序運行起來,就要調用resume函數。
如下代碼:

復制代碼 代碼如下:

    local co = coroutine.create(function() print("hello coroutine"); end);
    coroutine.resume(co);

輸出結果如下:

復制代碼 代碼如下:

[LUA-print] hello coroutine

3.更像樣的協同程序

剛剛那個協同程序太簡陋的,沒有任何作用,直接打印一條語句之后就結束了,同時它的狀態(tài)也變成了死亡狀態(tài)。

我們來一個帥一點的協同程序:

復制代碼 代碼如下:

    local co = coroutine.create(function()
        for i = 1, 2, 1 do
            print("木頭挺聰明的+" .. i);
        end
    end);
    coroutine.resume(co);

運行結果如下:

復制代碼 代碼如下:

[LUA-print] 木頭挺聰明的+1
[LUA-print] 木頭挺聰明的+2

所以我就說,電腦就是誠實,這日志打印的,真好看(小若:我們不要理這個神經病了)

4.讓協同程序掛起——yield

既然協同程序和線程差不多,那肯定不能讓協同程序一次過執(zhí)行完畢了,這就沒有意義了。
我們來看看怎么讓協同程序掛起,如下代碼:

復制代碼 代碼如下:

    local co = coroutine.create(function()
        for i = 1, 2, 1 do
            print("木頭挺聰明的+" .. i);
            coroutine.yield();
        end
    end);
    coroutine.resume(co);
    print(coroutine.status(co));

輸出結果如下:

復制代碼 代碼如下:

[LUA-print] 木頭挺聰明的+1
[LUA-print] suspended

這回就只輸出了一條日志就停止了,后面我們還調用了status函數,打印協同程序當前的狀態(tài),suspended即為掛起狀態(tài)。

因為這個協同程序還沒有執(zhí)行完畢,所以只能是掛起狀態(tài)。
 
那么,如果讓這協同程序繼續(xù)執(zhí)行呢?很簡單,再次調用resume函數,如代碼:

復制代碼 代碼如下:

local co = coroutine.create(function()
        for i = 1, 2, 1 do
            print("木頭挺聰明的+" .. i);
            coroutine.yield();
        end
    end);
    coroutine.resume(co);
    print(coroutine.status(co));
  
    coroutine.resume(co);
    print(coroutine.status(co));
  
    coroutine.resume(co);
    print(coroutine.status(co));

這次有點復雜了,先看看輸出結果:

復制代碼 代碼如下:

[LUA-print] 木頭挺聰明的+1
[LUA-print] suspended
[LUA-print] 木頭挺聰明的+2
[LUA-print] suspended
[LUA-print] dead

我一共執(zhí)行了三次resume函數,但很顯然,這個協同程序的for循環(huán)只會執(zhí)行2次。

那為什么第二次resume執(zhí)行之后,協同程序的狀態(tài)還是掛起呢?不應該是結束了么?結束了就應該是死亡狀態(tài)了。

而第三次執(zhí)行resume之后,反而沒有任何輸出,此時的狀態(tài)才真正切換到死亡狀態(tài)。
 
這是為什么呢?(小若:趕緊說,不說我看電影去了)
再來這么看看就明白了,加幾條打印代碼:

復制代碼 代碼如下:

local co = coroutine.create(function()
        for i = 1, 2, 1 do
            print("木頭挺聰明的+" .. i);
            coroutine.yield();
            print("一次循環(huán)結束");
        end
        print("協同程序結束");
    end);
    coroutine.resume(co);
    print(coroutine.status(co));
  
    coroutine.resume(co);
    print(coroutine.status(co));
  
    coroutine.resume(co);
    print(coroutine.status(co));

輸出結果如下:

復制代碼 代碼如下:

[LUA-print] 木頭挺聰明的+1
[LUA-print] suspended
[LUA-print] 一次循環(huán)結束
[LUA-print] 木頭挺聰明的+2
[LUA-print] suspended
[LUA-print] 一次循環(huán)結束
[LUA-print] 協同程序結束
[LUA-print] dead

這就很明顯了,在協同程序里調用yield函數時,會被掛起,而yield函數的返回要等下一次調用resume函數時才能得到。

所以,yield函數下面的print語句在下一次的resume調用時才被執(zhí)行。

又所以,當for循環(huán)第二次執(zhí)行時,協同程序被掛起,需要等待再一次resume時,for循環(huán)才能真正執(zhí)行完畢。

這就是這段代碼的特殊之處了。

5.resume操作的返回值

其實resume函數是有返回值的。
我們試試運行下面的代碼:

復制代碼 代碼如下:

    local co = coroutine.create(function()
        for i = 1, 2, 1 do
            coroutine.yield();
        end
    end);
    local result, msg = coroutine.resume(co);
    print(result);
    print(msg);

輸出結果如下:
復制代碼 代碼如下:

[LUA-print] true
[LUA-print] nil

resume返回兩個值,第一個值代表協同程序是否正常執(zhí)行,第二個返回值自然是代表錯誤信息。
我們試試讓協同程序出現錯誤:
復制代碼 代碼如下:

    local co = coroutine.create(function()
        error("呵呵,報錯了吧");
    end);
    local result, msg = coroutine.resume(co);
    print(result);
    print(msg);

輸出結果如下:

復制代碼 代碼如下:

[LUA-print] false
[LUA-print] [string "src/main.lua"]:91: 呵呵,報錯了吧

6.結束

好了,雖然我已經寫了這么多了,但是我真正想記錄的東西還沒開始寫呢~!
我了個噗,今晚我還能不能好好玩了…
好吧,內容有點多,下一篇繼續(xù)…

相關文章

  • Openresty服務器使用lua腳本寫的Hello World簡單實例

    Openresty服務器使用lua腳本寫的Hello World簡單實例

    這篇文章主要介紹了Openresty服務器使用lua腳本寫的Hello World簡單實例,OpenResty (也稱為 ngx_openresty)是一個全功能的 Web 應用服務器。它打包了標準的 Nginx 核心,很多的常用的第三方模塊,以及它們的大多數依賴項,需要的朋友可以參考下
    2015-04-04
  • Lua和C++的通信流程代碼實例

    Lua和C++的通信流程代碼實例

    這篇文章主要介紹了Lua和C++的通信流程代碼實例,本文是上一篇的DEMO,本文用代碼講解Lua和C++之間的通信,需要的朋友可以參考下
    2014-09-09
  • Lua中的協同程序之resume-yield間的數據返回研究

    Lua中的協同程序之resume-yield間的數據返回研究

    這篇文章主要介紹了Lua中的協同程序之resume-yield間的數據返回研究本文講解了resume的參數、resume函數的第二個返回值、yield的返回值等內容,需要的朋友可以參考下
    2014-09-09
  • Lua中對table排序實例

    Lua中對table排序實例

    這篇文章主要介紹了Lua中對table排序實例,本文講解了Lua中對table的一般排序方法、針對值的排序、同時對鍵值進行排序等方法,需要的朋友可以參考下
    2014-09-09
  • Lua獲取utf8字符串長度和字符串截取并用...代替

    Lua獲取utf8字符串長度和字符串截取并用...代替

    這篇文章主要介紹了Lua獲取utf8字符串長度和字符串截取并用...代替,本文直接給出兩個函數的實現代碼,需要的朋友可以參考下
    2015-04-04
  • 詳解Lua中repeat...until循環(huán)語句的使用方法

    詳解Lua中repeat...until循環(huán)語句的使用方法

    這篇文章主要介紹了詳解Lua中repeat...until循環(huán)語句的使用方法,需要的朋友可以參考下
    2015-05-05
  • Lua中的for循環(huán)和迭代器的秘密探究

    Lua中的for循環(huán)和迭代器的秘密探究

    這篇文章主要介紹了Lua中的for循環(huán)和迭代器的秘密探究,本文講解了返回兩個值的迭代器、for .. in .. do的真面目、迭代器函數、恒定狀態(tài)、控制變量初值等內容,需要的朋友可以參考下
    2014-09-09
  • Lua中算術運算符的使用示例

    Lua中算術運算符的使用示例

    這篇文章主要介紹了Lua中算術運算符的使用示例,是Lua入門學習中的基礎知識,需要的朋友可以參考下
    2015-05-05
  • Lua中的捕獲機制和轉換技巧介紹

    Lua中的捕獲機制和轉換技巧介紹

    這篇文章主要介紹了Lua中的捕獲機制和轉換技巧介紹,捕獲是這樣一種機制:可以使用模式串的一部分匹配目標串的一部分,將你想捕獲的模式用圓括號括起來,就指定了一個捕獲,需要的朋友可以參考下
    2014-11-11
  • Luvit像Node.js一樣寫Lua應用

    Luvit像Node.js一樣寫Lua應用

    今天小編就為大家分享一篇關于Luvit像Node.js一樣寫Lua應用,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2018-12-12

最新評論