OpenResty从入门到实战
温铭
OpenResty软件基金会主席,《OpenResty 最佳实践》作者
立即订阅
4332 人已学习
课程目录
已完结 52 讲
0/4登录后,你可以任选4讲全文学习。
开篇词 (1讲)
开篇词 | OpenResty,为你打开高性能开发的大门
免费
入门篇 (14讲)
01 | 初探OpenResty的三大特性
02 | 如何写出你的“hello world”?
03 | 揪出隐藏在背后的那些子项目
04 | 如何管理第三方包?从包管理工具luarocks和opm说起
05 | [视频]opm项目导读
06 | OpenResty 中用到的 NGINX 知识
07 | 带你快速上手 Lua
08 | LuaJIT分支和标准Lua有什么不同?
09 | 为什么 lua-resty-core 性能更高一些?
10 | JIT编译器的死穴:为什么要避免使用 NYI ?
11 | 剖析Lua唯一的数据结构table和metatable特性
12 | 高手秘诀:识别Lua的独有概念和坑
13 | [视频]实战:基于FFI实现的lua-resty-lrucache
14 | 答疑(一):Lua 规则和 NGINX 配置文件产生冲突怎么办?
API篇 (11讲)
15 | OpenResty 和别的开发平台有什么不同?
16 | 秒杀大多数开发问题的两个利器:文档和测试案例
17 | 为什么能成为更好的Web服务器?动态处理请求和响应是关键
18 | worker间的通信法宝:最重要的数据结构之shared dict
19 | OpenResty 的核心和精髓:cosocket
20 | 超越 Web 服务器:特权进程和定时任务
21 | 带你玩转时间、正则表达式等常用API
22 | [视频]从一个安全漏洞说起,探寻API性能和安全的平衡
23 | [视频]导读lua-resty-requests:优秀的lua-resty-*是如何编写的?
24 | 实战:处理四层流量,实现Memcached Server
25 | 答疑(二):特权进程的权限到底是什么?
测试篇 (5讲)
26 | 代码贡献者的拦路虎:test::nginx 简介
27 | test::nginx 包罗万象的测试方法
28 | test::nginx 还可以这样用?
29 | 最容易失准的性能测试?你需要压测工具界的“悍马”wrk
30 | 答疑(三)如何搭建测试的网络结构?
性能优化篇 (16讲)
31 | 性能下降10倍的真凶:阻塞函数
32 | 让人又恨又爱的字符串操作
33 | 性能提升10倍的秘诀:必须用好 table
34 | 特别放送:OpenResty编码指南
35 | [视频]实际项目中的性能优化:ingress-nginx中的几个PR解读
36 | 盘点OpenResty的各种调试手段
37 | systemtap-toolkit和stapxx:如何用数据搞定“疑难杂症”?
38 | [视频]巧用wrk和火焰图,科学定位性能瓶颈
39 | 高性能的关键:shared dict 缓存和 lru 缓存
40 | 缓存与风暴并存,谁说缓存风暴不可避免?
41 | lua-resty-* 封装,让你远离多级缓存之痛
42 | 如何应对突发流量:漏桶和令牌桶的概念
43 | 灵活实现动态限流限速,其实没有那么难
44 | OpenResty 的杀手锏:动态
45 | 不得不提的能力外延:OpenResty常用的第三方库
46 | 答疑(四):共享字典的缓存是必须的吗?
API网关篇 (4讲)
47 | 微服务API网关搭建三步曲(一)
48 | 微服务API网关搭建三步曲(二)
49 | 微服务API网关搭建三步曲(三)
50 | 答疑(五):如何在工作中引入 OpenResty?
结束语 (1讲)
结束语 | 行百里者半九十
OpenResty从入门到实战
登录|注册

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

温铭 2019-06-12
你好,我是温铭。
这节课,我们来学习下 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/1000字
划线
笔记
复制
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
该试读文章来自付费专栏《OpenResty从入门到实战》,如需阅读全部文章,
请订阅文章所属专栏。
立即订阅
登录 后留言

精选留言(14)

  • Jeff.Smile
    例子讲的还是挺清楚的,希望继续讲一下具体openresty实际项目的使用场景案例
    2019-06-12
    4
  • 空知
    resty -e "local t = {1, nil, name='cjf', 2} print(table.getn(t))"
    3

    老师 测试下这个 咋会出来3呢 不是遇见nil就停止了吗
    2019-06-13
    1
    3
  • 轨迹
    老师可以讲一下table的内部结构,把原理搞清楚,就比较容易了。
    譬如这个文章:https://blog.csdn.net/wwlcsdn000/article/details/81291756
    简单的就是两点:
    1、两种存储,哈希和数组。
    2、数组以下标覆盖到哈希,如果遇到key冲突,数组覆盖哈希的value。
    我个人感觉就更好一点。

    作者回复: 多谢补充 :)

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

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

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

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

    2019-06-12
    2
  • 岁月如歌
    @空知 提出的问题 有相同的疑问?

    resty -e "local t = {1, nil, name='cjf', 2} print(table.getn(t))"
    3

    老师 测试下这个 咋会出来3呢 不是遇见nil就停止了吗

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

    2019-07-13
    1
  • HelloTalk
    原来默认写lua 的时候,需要做二进制 位移、抑或的操作,为此自己写了一个bitop的库,发现性能更不上,后面上了 luajit,测试快了很多。
    2019-10-20
  • KoALa
    lua if语句的判断方式和ruby貌似一样,nil和false才为假,其他都为真,用多了个人觉得这种判断方式更合理~~

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

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

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

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

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

    2019-06-15
    1
  • shineyyl
    $ resty -e 'local t; t={100}; ngx.say(t[1]);print(type(t[1]))'
    100
    number
    $ resty -e 'local t; t={100}; ngx.say(t[0]);print(type(t[0]))'
           #空
    nil #nil,需要print打印出来t的数据类型
    2019-06-12
  • 一步
    老师 文中说了 序列不应该含有 nil的,array 是可以包含 nil
    2019-06-12
  • 旺旺
    1."在同一个 worker 内的所有协程",协程就是常说的线程吧?
    2.$ resty -e 't={100}; ngx.say(t[0])'
    这代码,变量定义前面需要加local才能运行。
    3.“对应到上面的代码中,除了 t2 外,其他的 table 都是 array。”
    这句话写反了吧,应该是“除了 t2 外,其他的 table 都是 序列。”吧
    2019-06-12
收起评论
14
返回
顶部