哎,周五晚上我都還這么努力看書,真是好孩子。(小若:不想吐槽了)
其實我都準備玩游戲看電影去的了,但是這書就擺在桌子上,而且正對著我,就想著,掃兩眼吧。
結果一掃就不對勁了,因為這內容有點繞,有點小混亂,如果我現在不記錄下來的話,下周一可能又要重新看一次了。
好吧,今天我們來聊聊協同程序。
1.什么是協同程序(coroutinue)
大家都知道線程吧?都知道多線程吧?協同程序就和這線程差不多,但是又有比較明顯的區別。
多個協同程序在任意時刻只能執行一個,雖然線程在某種意義上也是這樣,但這不是一樣的概念。
換句話說,一個協同程序在運行的時候,其他協同程序是無法獲得執行的機會的。
只有正在運行的協同程序主動掛起時,其他協同程序才有機會執行。
而線程呢?即使不主動休眠,也很有可能因為輪片時間到達而把執行機會讓給其他線程。
2.創建協同程序
創建協同程序很簡單,咋一看,其實和線程沒差別~
代碼如下:
復制代碼 代碼如下:
local co = coroutine.create(function() print("hello coroutine"); end);
協同的程序的操作都在coroutine里,create函數的參數就是協同程序要執行的函數,就這么運行代碼是沒有效果的。
因為協同程序創建后,默認是掛起狀態。
協同程序的四種狀態分別為:掛起(suspended)、運行(running)、死亡(dead)、正常(normal)。
要想協同程序運行起來,就要調用resume函數。
如下代碼:
復制代碼 代碼如下:
local co = coroutine.create(function() print("hello coroutine"); end);
coroutine.resume(co);
輸出結果如下:
復制代碼 代碼如下:
[LUA-print] hello coroutine
3.更像樣的協同程序
剛剛那個協同程序太簡陋的,沒有任何作用,直接打印一條語句之后就結束了,同時它的狀態也變成了死亡狀態。
我們來一個帥一點的協同程序:
復制代碼 代碼如下:
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
既然協同程序和線程差不多,那肯定不能讓協同程序一次過執行完畢了,這就沒有意義了。
我們來看看怎么讓協同程序掛起,如下代碼:
復制代碼 代碼如下:
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函數,打印協同程序當前的狀態,suspended即為掛起狀態。
因為這個協同程序還沒有執行完畢,所以只能是掛起狀態。
那么,如果讓這協同程序繼續執行呢?很簡單,再次調用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
我一共執行了三次resume函數,但很顯然,這個協同程序的for循環只會執行2次。
那為什么第二次resume執行之后,協同程序的狀態還是掛起呢?不應該是結束了么?結束了就應該是死亡狀態了。
而第三次執行resume之后,反而沒有任何輸出,此時的狀態才真正切換到死亡狀態。
這是為什么呢?(小若:趕緊說,不說我看電影去了)
再來這么看看就明白了,加幾條打印代碼:
復制代碼 代碼如下:
local co = coroutine.create(function()
for i = 1, 2, 1 do
print("木頭挺聰明的+" .. i);
coroutine.yield();
print("一次循環結束");
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] 一次循環結束
[LUA-print] 木頭挺聰明的+2
[LUA-print] suspended
[LUA-print] 一次循環結束
[LUA-print] 協同程序結束
[LUA-print] dead
這就很明顯了,在協同程序里調用yield函數時,會被掛起,而yield函數的返回要等下一次調用resume函數時才能得到。
所以,yield函數下面的print語句在下一次的resume調用時才被執行。
又所以,當for循環第二次執行時,協同程序被掛起,需要等待再一次resume時,for循環才能真正執行完畢。
這就是這段代碼的特殊之處了。
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返回兩個值,第一個值代表協同程序是否正常執行,第二個返回值自然是代表錯誤信息。
我們試試讓協同程序出現錯誤:
復制代碼 代碼如下:
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.結束
好了,雖然我已經寫了這么多了,但是我真正想記錄的東西還沒開始寫呢~!
我了個噗,今晚我還能不能好好玩了…
好吧,內容有點多,下一篇繼續…
您可能感興趣的文章:- Lua協程(coroutine)程序運行分析
- Lua的協程(coroutine)簡介
- Lua之協同程序coroutine代碼實例
- Lua協同程序(COROUTINE)運行步驟分解
- Lua協同程序函數coroutine使用實例
- Lua編程示例(七):協同程序基礎邏輯
- 舉例詳解Lua中的協同程序編程
- Lua中的協同程序詳解
- Lua中的協同程序之resume-yield間的數據返回研究
- Lua協同程序coroutine的簡介及優缺點