背景
- 最近研究网关,逛了一大圈,由于部门的主打语言是golang和py所有尽量找golang圈的,但是go圈的网关都不太成熟。唯一好一点的便是tyk,但是tyk商业化比较严重,各种license,后面逛到openresty ,看起来诱惑力比较大,而且是基于nginx的,没的说,nginx经过各大公司验证了的,所以性能没得说,后面又看到了kong,在openresty的基础上做了很多事情,非常符合我们部门的需要,即需要在非常短的时间内出一套东西来,并且能达到基本的功能,kong的各种feature大多是我们需要的,我们只需要进行不大的二次开发变能变成一个成熟的东西,但在这个基础上涉及到另一门语言便是lua,下面介绍下在openresty下的lua。
介绍
- Lua 是一个小巧的脚本语言。
openresty采用的是LuaJIT 2 ,而不是标准的lua
。
- 解释下什么是jit
- LuaJIT 的运行时环境包括一个用手写汇编实现的 Lua 解释器和一个可以直接生成机器代码的 JIT 编译器。
Lua 代码在被执行之前总是会先被 lfn 成 LuaJIT 自己定义的字节码(Byte Code)。关于 LuaJIT 字节码的文档,可以参见:http://wiki.luajit.org/Bytecode-2.0(这个文档描述的是 LuaJIT 2.0 的字节码,不过 2.1 里面的变化并不算太大)。
一开始的时候,Lua 字节码总是被 LuaJIT 的解释器解释执行。LuaJIT 的解释器会在执行字节码时同时记录一些运行时的统计信息,比如每个 Lua 函数调用入口的实际运行次数,还有每个 Lua 循环的实际执行次数。当这些次数超过某个预设的阈值时,便认为对应的 Lua 函数入口或者对应的 Lua 循环足够的“热”,这时便会触发 JIT 编译器开始工作。
JIT 编译器会从热函数的入口或者热循环的某个位置开始尝试编译对应的 Lua 代码路径。编译的过程是把 LuaJIT 字节码先转换成 LuaJIT 自己定义的中间码(IR),然后再生成针对目标体系结构的机器码(比如 x86_64 指令组成的机器码)。
- 当然jit并不是支持所有的原语,所有对性能要求很高的情况下,应该尽量使用jit支持的原语。
环境搭建
- 按照官网的搭建方式即可
- eg: mac 上环境搭建
tar zxf LuaJIT-2.0.5.tar.gz
cd LuaJIT-2.0.5
make
make install
数据类型
- nil、boolean、 number、string、table、function
* lua将nil表示为`无效值` ,一个变量在第一次赋值前默认为nil,将 nil 赋予给一个全局变量就等同于删除它。
* Lua 中 nil 和 false 为“假”,其它所有值均为“真”。比如 0 和空字符串就是“真”
* 一般地,Lua 的 number 类型就是用双精度浮点数来实现的。值得一提的是,LuaJIT 支持所谓的“dual-number”(双数)模式,即 LuaJIT 会根据上下文用整型来存储整数,而用双精度浮点数来存放浮点数
* 三种写法: 单引号、双引号、长括号([[]])
* Table 类型实现了一种抽象的“关联数组”。“关联数组”是一种具有特殊索引方式的数组,索引通常是字符串(string)或者 number 类型,但也可以是除 nil 以外的任意类型的值。
* 在 Lua 中,函数 也是一种数据类型,函数可以存储在变量中,可以通过参数传递给其他函数,还可以作为其他函数的返回值。
表达式(着重说下不一样)
- 算术运算符
- print(5.0 / 10) -->打印 0.5。
浮点数相除的结果是浮点数
- 关系运算符
- 逻辑运算符
控制结构
分支结构 (elseif
)
if x > 0 then
print("x is a positive number")
end
if x > 0 then
print("x is a positive number")
else
print("x is a non-positive number")
end
if score == 100 then
print("Very good!Your score is 100")
elseif score >= 60 then
print("Congratulations, you have passed it,your score greater or equal to 60")
--此处可以添加多个elseif
else
print("Sorry, you do not pass the exam! ")
end
循环结构(没有提供 continue,却也提供了另外一个标准控制语句 break
)
while 表达式 do
--body
end
x = 10
repeat
print(x)
until false
for var = begin, finish, step do
--body
end
只能遍历数组
local a = {"a", "b", "c", "d"}
for i, v in ipairs(a) do
print("index:", i, " value:", v)
end
遍历 k、v
for k,v in pairs(revDays) do
print("k:", k, " v:", v)
end
ipairs() 内建函数是可以被 JIT 编译的,而 pairs() 则只能被解释执行