主頁 > 知識(shí)庫 > Lua中的協(xié)同程序探究

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

熱門標(biāo)簽:如何利用高德地圖標(biāo)注家 上海企業(yè)外呼系統(tǒng)排名 開通400電話申請(qǐng)流程 400手機(jī)電話免費(fèi)辦理 武漢百應(yīng)人工智能電銷機(jī)器人 電腦外呼系統(tǒng)輻射大嗎 揚(yáng)州電銷外呼系統(tǒng)軟件 百度地圖標(biāo)注位置網(wǎng)站 智能語音電銷的機(jī)器人

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

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

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

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

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

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

換句話說,一個(gè)協(xié)同程序在運(yùn)行的時(shí)候,其他協(xié)同程序是無法獲得執(zhí)行的機(jī)會(huì)的。
只有正在運(yùn)行的協(xié)同程序主動(dòng)掛起時(shí),其他協(xié)同程序才有機(jī)會(huì)執(zhí)行。
 
而線程呢?即使不主動(dòng)休眠,也很有可能因?yàn)檩喥瑫r(shí)間到達(dá)而把執(zhí)行機(jī)會(huì)讓給其他線程。

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

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

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

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

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

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

協(xié)同程序的四種狀態(tài)分別為:掛起(suspended)、運(yùn)行(running)、死亡(dead)、正常(normal)。
 
要想?yún)f(xié)同程序運(yùn)行起來,就要調(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é)同程序

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

我們來一個(gè)帥一點(diǎn)的協(xié)同程序:

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

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

運(yùn)行結(jié)果如下:

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

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

所以我就說,電腦就是誠(chéng)實(shí),這日志打印的,真好看(小若:我們不要理這個(gè)神經(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)。

因?yàn)檫@個(gè)協(xié)同程序還沒有執(zhí)行完畢,所以只能是掛起狀態(tài)。
 
那么,如果讓這協(xié)同程序繼續(xù)執(zhí)行呢?很簡(jiǎn)單,再次調(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));

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

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

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

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

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

而第三次執(zhí)行resume之后,反而沒有任何輸出,此時(shí)的狀態(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ù)時(shí),會(huì)被掛起,而yield函數(shù)的返回要等下一次調(diào)用resume函數(shù)時(shí)才能得到。

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

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

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

5.resume操作的返回值

其實(shí)resume函數(shù)是有返回值的。
我們?cè)囋囘\(yùn)行下面的代碼:

復(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返回兩個(gè)值,第一個(gè)值代表協(xié)同程序是否正常執(zhí)行,第二個(gè)返回值自然是代表錯(cuò)誤信息。
我們?cè)囋囎寘f(xié)同程序出現(xiàn)錯(cuò)誤:
復(fù)制代碼 代碼如下:

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

輸出結(jié)果如下:

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

[LUA-print] false
[LUA-print] [string "src/main.lua"]:91: 呵呵,報(bào)錯(cuò)了吧

6.結(jié)束

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

您可能感興趣的文章:
  • Lua協(xié)程(coroutine)程序運(yùn)行分析
  • Lua的協(xié)程(coroutine)簡(jiǎn)介
  • Lua之協(xié)同程序coroutine代碼實(shí)例
  • Lua協(xié)同程序(COROUTINE)運(yùn)行步驟分解
  • Lua協(xié)同程序函數(shù)coroutine使用實(shí)例
  • Lua編程示例(七):協(xié)同程序基礎(chǔ)邏輯
  • 舉例詳解Lua中的協(xié)同程序編程
  • Lua中的協(xié)同程序詳解
  • Lua中的協(xié)同程序之resume-yield間的數(shù)據(jù)返回研究
  • Lua協(xié)同程序coroutine的簡(jiǎn)介及優(yōu)缺點(diǎn)

標(biāo)簽:武漢 黑龍江 宜賓 延邊 張掖 新余 江西 嘉峪關(guān)

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Lua中的協(xié)同程序探究》,本文關(guān)鍵詞  Lua,中的,協(xié)同,程序,探究,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Lua中的協(xié)同程序探究》相關(guān)的同類信息!
  • 本頁收集關(guān)于Lua中的協(xié)同程序探究的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章