OpenResty 从入门到实战
温铭
OpenResty 软件基金会第一任主席,Apache APISIX 项目 VP
20903 人已学习
新⼈⾸单¥59
登录后,你可以任选4讲全文学习
课程目录
已完结/共 52 讲
结束语 (1讲)
OpenResty 从入门到实战
15
15
1.0x
00:00/00:00
登录|注册

08 | LuaJIT分支和标准Lua有什么不同?

写在最后
LuaJIT
Lua 特别之处
为什么选择 LuaJIT?
标准 Lua 和 LuaJIT 的关系
LuaJIT 分支和标准Lua有什么不同?
文章主题总结

该思维导图由 AI 生成,仅供参考

你好,我是温铭。
这节课,我们来学习下 OpenResty 的另一块基石:LuaJIT。今天主要的篇幅,我会留给 Lua 和 LuaJIT 中重要和鲜为人知的一些知识点。而更多 Lua 语言的基础知识,你可以通过搜索引擎或者 Lua 的书籍自己来学习,这里我推荐 Lua 作者编写的《Lua 程序设计》这本书。
当然,在 OpenResty 中,写出正确的 LuaJIT 代码的门槛并不高,但要写出高效的 LuaJIT 代码绝非易事,这里的关键内容,我会在后面 OpenResty 性能优化部分详细介绍。
我们先来看下 LuaJIT 在 OpenResty 整体架构中的位置:
前面我们提到过,OpenResty 的 worker 进程都是 fork master 进程而得到的, 其实, master 进程中的 LuaJIT 虚拟机也会一起 fork 过来。在同一个 worker 内的所有协程,都会共享这个 LuaJIT 虚拟机,Lua 代码的执行也是在这个虚拟机中完成的。
这可以算是 OpenResty 的基本原理,后面课程我们再详细聊聊。今天我们先来理顺 Lua 和 LuaJIT 的关系。

标准 Lua 和 LuaJIT 的关系

先把重要的事情放在前面说:
标准 Lua 和 LuaJIT 是两回事儿,LuaJIT 只是兼容了 Lua 5.1 的语法。
确认放弃笔记?
放弃后所记笔记将不保留。
新功能上线,你的历史笔记已初始化为私密笔记,是否一键批量公开?
批量公开的笔记不会为你同步至部落
公开
同步至部落
取消
完成
0/2000
荧光笔
直线
曲线
笔记
复制
AI
  • 深入了解
  • 翻译
    • 英语
    • 中文简体
    • 中文繁体
    • 法语
    • 德语
    • 日语
    • 韩语
    • 俄语
    • 西班牙语
    • 阿拉伯语
  • 解释
  • 总结

LuaJIT是OpenResty的关键组成部分,与标准Lua有着显著的区别。LuaJIT是Lua 5.1语法的兼容版本,而标准Lua的最新版本是5.3。在OpenResty中,已经去掉了对标准Lua的支持,只支持LuaJIT。LuaJIT的性能优势主要在于其JIT编译器能够直接生成机器码,而不是回退到Lua解释器的解释执行模式。此外,LuaJIT还紧密结合了FFI(Foreign Function Interface),可以让你直接在Lua代码中调用外部的C函数和使用C的数据结构。对于性能优化,LuaJIT还扩展了table的相关函数:`table.new`和`table.clear`。这些特性使得LuaJIT成为OpenResty性能优化的基础,对于后续学习OpenResty性能优化至关重要。总的来说,选择LuaJIT的主要原因是其性能优势。文章还提到了Lua语言的特别之处,包括下标从1开始、使用`..`来拼接字符串、只有table一种数据结构以及默认是全局变量。在实际项目开发中,这些特点可能会导致一些意料之外的结果,需要开发者注意。文章还提到了FFI的性能优势以及LuaJIT扩展的table函数,这些内容对于开发者来说具有实际的指导意义。

仅可试看部分内容,如需阅读全部内容,请付费购买文章所属专栏
《OpenResty 从入门到实战》
新⼈⾸单¥59
立即购买
登录 后留言

全部留言(20)

  • 最新
  • 精选
  • 轨迹
    老师可以讲一下table的内部结构,把原理搞清楚,就比较容易了。 譬如这个文章:https://blog.csdn.net/wwlcsdn000/article/details/81291756 简单的就是两点: 1、两种存储,哈希和数组。 2、数组以下标覆盖到哈希,如果遇到key冲突,数组覆盖哈希的value。 我个人感觉就更好一点。

    作者回复: 多谢补充 :)

    2019-06-12
    19
  • 逍遥
    openresty为什么要维护自己的luajit分支呢,为什么不能用luajit

    作者回复: 有两个原因:一个是 LuaJIT 的作者基本处于退休状态,只修 bug,不怎么加新功能,关于 LuaJIT 的 bug,OpenResty 还是会提交给LuaJIT 官方的;第二个是新增的主要是 OpenResty 优化中遇到的 API,自己维护更容易控制版本和节奏。

    2019-07-08
    5
  • 上面说的 luajit 先把 字节码转为中间码爱转为 机器码 这个有个以为你,字节码不就是机器码吗? 不都是二进制的东西吗

    作者回复: 这里的字节码是 LuaJIT 虚拟机执行的一种指令格式;机器码是指 CPU 可以读取的指令格式。

    2019-06-12
    3
  • HelloBug
    有遇到一些让人困惑的地方是ngx.null、nil、null、“”。在网上搜索的时候,有看到说null是ngx.null的一个定义。redis的返回值的时候,经常会判断返回结果是否为空,判断的时候是和哪个值进行比较呢?关于这些个值有没有其他一些使用上的坑呢?一直以来都没有有一个明确的认识,所以和老师确认一下。

    作者回复: 在 lua-resty-redis 里面,查找一个key 的时候: local res, err = red:get("dog") 如果返回值 res 是 nil,就说明函调用失败了;如果 res 是 ngx.null 就说明redis 中不存在 dog 这个key。 而在处理 cjson 的时候,又有cjson.null这个值。 所以还是要根据对应库的文档来做空值的判断和区分。 在写类似 if not res then 这样的代码的时候,要特别留意下,最好改成明确的: if res ~= nil and res ~= false then ,类似这样的,并有对应的测试案例覆盖。

    2019-06-15
    2
  • 岁月如歌
    @空知 提出的问题 有相同的疑问? 【 resty -e "local t = {1, nil, name='cjf', 2} print(table.getn(t))" 3 老师 测试下这个 咋会出来3呢 不是遇见nil就停止了吗 】

    作者回复: 我建议大家绕着走,把 nil 改为 ngx.null 来填充数组。不同的 lua 版本会有不同的行为,我也不太清楚。

    2019-07-13
    5
    1
  • HelloBug
    温铭老师,你好~ 在“为什么选择luaJIT”的讲解中,讲到如果lua函数入口或者lua循环足够热,JIT编译器会从热函数的入口或者热循环的某个位置开始尝试编译。有以下几个问题请教一下老师。 1.这里的“尝试”编译的意思是并不是所有的热函数或者热循环都可以被编译,是吧? 2.另一个疑问是为什么一开始不对所有的可以被JIT编译的函数或者循环直接进行编译呢? 3.现在的JIT发现热点然后进行编译,为什么判断是否是热点时的统计,是根据一个随机的阈值,而不是一个其他,比如经过测试认为比较合理的值?

    作者回复: LuaJIT 是会记录并统计代码的运行次数,才能知道哪些是热函数和热循环,这就是“尝试”的含义; 如果有些函数只调用了一两次,就没有编译的必要性了; 第三个问题,我的猜测可能是处于安全或者类似的考虑,就没有写死一个值,而是随机。

    2019-06-15
    2
    1
  • KoALa
    lua if语句的判断方式和ruby貌似一样,nil和false才为假,其他都为真,用多了个人觉得这种判断方式更合理~~

    作者回复: 主要是OpenResty 中空值的情况比较多

    2019-07-24
  • 杨丁
    内容通俗易懂,赞。老师什么时候出个限流的实战吧

    作者回复: 后面有专门的限流限速章节

    2019-07-12
  • 空知
    resty -e "local t = {1, nil, name='cjf', 2} print(table.getn(t))" 3 老师 测试下这个 咋会出来3呢 不是遇见nil就停止了吗
    2019-06-13
    5
    8
  • J.Smile
    例子讲的还是挺清楚的,希望继续讲一下具体openresty实际项目的使用场景案例
    2019-06-12
    5
收起评论
显示
设置
留言
20
收藏
沉浸
阅读
分享
手机端
快捷键
回顶部