codecamp

Lua Web快速开发指南(2) - cf的运行机制简介与基于httpd库的开发环境搭建

从本章开始假设大家都熟悉lua语法.

运行流程

在上一章节学会了如何安装cf后, 本章节就会介绍cf到运行机制与httpd的server搭建!

cf是一个非常典型的基于协程的事件驱动开发框架在封装成API后, 可以依赖事件循环执行一套稳定运行时环境.

而lua本身提供了强大的table数据结构可以根据需要自行构建所谓的"config", 所以cf为了减少无用的依赖就没有提供额外的config文件进行配置.

cf会假设所有业务代码文件都在script目录下, 所以建议您在script下自行划分好文件的目录归属.

cf将所有业务编写的脚本目录命名为script, script目录下点main.lua文件将会是入口文件. 这个main.lua执行完毕之后才会真正进入事件循环.

在执行完成script/main.lua文件后, cf则会是实际情况决定是否需要开始运行事件循环. 假设开发者仅仅想运行print("hello world"), 那么cf将会在main.lua执行完成后直接退出.

运行机制的问题与模块化的设计

main.lua文件内部可以导入其它lua代码进行模块化设计, 但是有一点需要注意的是: cf框架在使用require导入文件时, 导入的文件只能用于定义不可用于流程执行.

什么意思呢? 例如, 当你在使用创建一个httpd实例之前会先创建DB对象并且进行初始化! 这时候您可以回设计像下面的代码:

  -- 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", 这个错误提示的中文大概意思就是: "不能在非协程的环境下切换执行权".

这是因为require函数的对指定的脚本调用的Call方法使其不允许临时切换出执行权, 而初始化数据库连接操作是依赖cf的协程与异步操作所以就会导致上面的错误出现.

可以将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的内部call调用, require返回后执行权重新回到了由cf启动的协程中. 这样就可以开始正确初始化了.

这种情况仅限于依赖require执行某段异步代码块的时候, 其它情况下一般不会出现问题. 优质的项目管理人员一般不会出现这样的设计.

使用httpd库快速搭建lua web开发环境

httpd库是cf内置的基于http 1.1协议开发的web server! 高效解析器是必不可少的, httpd库使用picohttpparser解析器来构建http context.

我们假设您至少看过httpd库的API Reference, 并且至少知道下面所述的API.

此处所有的API与使用方式都将会在API Reference中找到.

1. 导入http库

httpd库位于app/lualib下, 使用者可以在main.lua文件内直接使用local httpd = require "httpd"导入httpd库.

2. 初始化一个httpd app对象

httpd库使用lua class对象进行创建! 默认提供了new方法, 使用者可以使用new方法创建一个httpd的app实例.

  local httpd = require "httpd"


  local app = httpd:new("app")

3. 注册静态文件路径

httpd提供给了内置的静态文件查找能力, 只需要使用者自动使用static方法注册静态文件路径即可.

  app:static("static", 30)

static表示使用者想将app/static文件夹当做静态文件的根目录.

4. 设置监听端口

httpd启动需要指定监听的端口, 默认监听所有网卡. 虽然没有使用第一个参数, 但是不可为空.

  app:listen("0.0.0.0", 8080)

5. 运行

在初始化完成后, app调用run方法将会启动httpd服务器. run方法后面的代码可能永远不会有机会执行.

  app:run()

6. 运行cf

使用./cfadmin命令运行httpd server, 如果您看到类似运行等字样说明httpd服务已经启动完成, 否则将会有响应的错误提示.

7. 完整的代码示例

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

体验一下

现在, 让我们打开http://localhost:8080/index.html查看是否能正确显示页面了呢?

更多的httpd API

这里你可以找到更多有关httpd库的API说明.

继续学习

一下章我们一起学习如何用cf来注册注册路由

Lua Web快速开发指南(1) - 初识cf框架
Lua Web快速开发指南(3) - 初识httpd库路由
温馨提示
下载编程狮App,免费阅读超1000+编程语言教程
取消
确定
目录

关闭

MIP.setData({ 'pageTheme' : getCookie('pageTheme') || {'day':true, 'night':false}, 'pageFontSize' : getCookie('pageFontSize') || 20 }); MIP.watch('pageTheme', function(newValue){ setCookie('pageTheme', JSON.stringify(newValue)) }); MIP.watch('pageFontSize', function(newValue){ setCookie('pageFontSize', newValue) }); function setCookie(name, value){ var days = 1; var exp = new Date(); exp.setTime(exp.getTime() + days*24*60*60*1000); document.cookie = name + '=' + value + ';expires=' + exp.toUTCString(); } function getCookie(name){ var reg = new RegExp('(^| )' + name + '=([^;]*)(;|$)'); return document.cookie.match(reg) ? JSON.parse(document.cookie.match(reg)[2]) : null; }