從本章開始假設(shè)大家都熟悉lua語法.
在上一章節(jié)學(xué)會了如何安裝cf后, 本章節(jié)就會介紹cf到運行機制與httpd的server搭建!
cf是一個非常典型的基于協(xié)程的事件驅(qū)動開發(fā)框架在封裝成API后, 可以依賴事件循環(huán)執(zhí)行一套穩(wěn)定運行時環(huán)境.
而lua本身提供了強大的table
數(shù)據(jù)結(jié)構(gòu)可以根據(jù)需要自行構(gòu)建所謂的"config", 所以cf為了減少無用的依賴就沒有提供額外的config文件進行配置.
cf會假設(shè)所有業(yè)務(wù)代碼文件都在script
目錄下, 所以建議您在script
下自行劃分好文件的目錄歸屬.
cf將所有業(yè)務(wù)編寫的腳本目錄命名為script
, script
目錄下點main.lua
文件將會是入口文件. 這個main.lua
執(zhí)行完畢之后才會真正進入事件循環(huán).
在執(zhí)行完成script/main.lua
文件后, cf則會是實際情況決定是否需要開始運行事件循環(huán). 假設(shè)開發(fā)者僅僅想運行print("hello world")
, 那么cf將會在main.lua
執(zhí)行完成后直接退出.
main.lua
文件內(nèi)部可以導(dǎo)入其它lua代碼進行模塊化設(shè)計, 但是有一點需要注意的是: cf框架在使用require
導(dǎo)入文件時, 導(dǎo)入的文件只能用于定義不可用于流程執(zhí)行.
什么意思呢? 例如, 當(dāng)你在使用創(chuàng)建一個httpd實例之前會先創(chuàng)建DB對象并且進行初始化! 這時候您可以回設(shè)計像下面的代碼:
-- mydb.lua
local DB = require "DB"
local db = DB:new {
host = "localhost",
port = 3306,
database = "cfadmin",
charset = 'utf8'
-- ...
}
local ok = db:connect()
if not ok then
return nil
end
return db
-- main.lua
local db_cls = require "mydb"
--[[
... do your want do
]]
這樣做你會得到一個錯誤提示:"attempt to yield from outside a coroutine", 這個錯誤提示的中文大概意思就是: "不能在非協(xié)程的環(huán)境下切換執(zhí)行權(quán)".
這是因為require函數(shù)的對指定的腳本調(diào)用的Call方法使其不允許臨時切換出執(zhí)行權(quán), 而初始化數(shù)據(jù)庫連接操作是依賴cf的協(xié)程與異步操作所以就會導(dǎo)致上面的錯誤出現(xiàn).
可以將mydb.lua與main.lua的代碼替換成下面這樣(Cache庫同理)來解決:
-- mydb.lua
local DB = require "DB"
local db
return function ()
if db then
return db
end
db = DB:new {
host = "localhost",
port = 3306,
database = "cfadmin",
charset = 'utf8'
-- ...
}
local ok = db:connect()
if not ok then
db:close()
db = nil
return nil, "連接失敗"
end
return db
end
-- main.lua
local get_db = require "mydb"
local db = get_db()
-- [[
... do your want do
]]
這是用一種巧妙的function與upvalue的方式避開了require的內(nèi)部call調(diào)用, require返回后執(zhí)行權(quán)重新回到了由cf啟動的協(xié)程中. 這樣就可以開始正確初始化了.
這種情況僅限于依賴require執(zhí)行某段異步代碼塊的時候, 其它情況下一般不會出現(xiàn)問題. 優(yōu)質(zhì)的項目管理人員一般不會出現(xiàn)這樣的設(shè)計.
httpd庫是cf內(nèi)置的基于http 1.1協(xié)議開發(fā)的web server! 高效解析器是必不可少的, httpd庫使用picohttpparser解析器來構(gòu)建http context.
我們假設(shè)您至少看過httpd庫的API Reference, 并且至少知道下面所述的API.
此處所有的API與使用方式都將會在API Reference中找到.
httpd庫位于app/lualib
下, 使用者可以在main.lua
文件內(nèi)直接使用local httpd = require "httpd"
導(dǎo)入httpd庫.
httpd庫使用lua class對象進行創(chuàng)建! 默認(rèn)提供了new方法, 使用者可以使用new方法創(chuàng)建一個httpd的app實例.
local httpd = require "httpd"
local app = httpd:new("app")
httpd提供給了內(nèi)置的靜態(tài)文件查找能力, 只需要使用者自動使用static方法注冊靜態(tài)文件路徑即可.
app:static("static", 30)
static表示使用者想將app/static
文件夾當(dāng)做靜態(tài)文件的根目錄.
httpd啟動需要指定監(jiān)聽的端口, 默認(rèn)監(jiān)聽所有網(wǎng)卡. 雖然沒有使用第一個參數(shù), 但是不可為空.
app:listen("0.0.0.0", 8080)
在初始化完成后, app調(diào)用run
方法將會啟動httpd服務(wù)器. run
方法后面的代碼可能永遠不會有機會執(zhí)行.
app:run()
使用./cfadmin
命令運行httpd server, 如果您看到類似運行
等字樣說明httpd服務(wù)已經(jīng)啟動完成, 否則將會有響應(yīng)的錯誤提示.
-- script/main.lua
local httpd = require "httpd"
local app = httpd:new("app")
app:static("static", 30)
app:listen("0.0.0.0", 8080)
app:run()
現(xiàn)在, 讓我們打開http://localhost:8080/index.html
查看是否能正確顯示頁面了呢?
在這里你可以找到更多有關(guān)httpd庫的API說明.
一下章我們一起學(xué)習(xí)如何用cf來注冊注冊路由
更多建議: