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

Lua中的協(xié)同程序探究

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

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

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

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

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

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

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

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

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

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

復(fù)制代碼 代碼如下:

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

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

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

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

復(fù)制代碼 代碼如下:

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

輸出結(jié)果如下:

復(fù)制代碼 代碼如下:

[LUA-print] hello coroutine

3.更像樣的協(xié)同程序

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

我們來一個帥一點的協(xié)同程序:

復(fù)制代碼 代碼如下:

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

運行結(jié)果如下:

復(fù)制代碼 代碼如下:

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

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

4.讓協(xié)同程序掛起——yield

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

復(fù)制代碼 代碼如下:

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

輸出結(jié)果如下:

復(fù)制代碼 代碼如下:

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

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

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

復(fù)制代碼 代碼如下:

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));

這次有點復(fù)雜了,先看看輸出結(jié)果:

復(fù)制代碼 代碼如下:

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

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

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

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

復(fù)制代碼 代碼如下:

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

輸出結(jié)果如下:

復(fù)制代碼 代碼如下:

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

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

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

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

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

5.resume操作的返回值

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

復(fù)制代碼 代碼如下:

    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);

輸出結(jié)果如下:
復(fù)制代碼 代碼如下:

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

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

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

輸出結(jié)果如下:

復(fù)制代碼 代碼如下:

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

6.結(jié)束

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

相關(guān)文章

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

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

    這篇文章主要介紹了Lua獲取utf8字符串長度和字符串截取并用...代替,本文直接給出兩個函數(shù)的實現(xiàn)代碼,需要的朋友可以參考下
    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的真面目、迭代器函數(shù)、恒定狀態(tài)、控制變量初值等內(nèi)容,需要的朋友可以參考下
    2014-09-09
  • Lua中算術(shù)運算符的使用示例

    Lua中算術(shù)運算符的使用示例

    這篇文章主要介紹了Lua中算術(shù)運算符的使用示例,是Lua入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下
    2015-05-05
  • Lua中的捕獲機(jī)制和轉(zhuǎn)換技巧介紹

    Lua中的捕獲機(jī)制和轉(zhuǎn)換技巧介紹

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

    Luvit像Node.js一樣寫Lua應(yīng)用

    今天小編就為大家分享一篇關(guān)于Luvit像Node.js一樣寫Lua應(yīng)用,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2018-12-12
  • 最新評論