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从入门到实战
登录|注册

21 | 带你玩转时间、正则表达式等常用API

温铭 2019-07-12
你好,我是温铭。在前面几节课中,你已经熟悉了不少 OpenResty 中重要的 Lua API 了,今天我们再来了解下其他一些通用的 API,主要和正则表达式、时间、进程等相关。

正则

先来看下最常用,也是最重要的正则。在 OpenResty 中,我们应该一直使用 ngx.re.* 提供的一系列 API,来处理和正则表达式相关的逻辑,而不是用 Lua 自带的模式匹配。这不仅是出于性能方面的考虑,还因为 Lua 自带的正则是自成体系的,并非 PCRE 规范,这对于绝大部分开发者来说都是徒增烦恼。
在前面的课程中,你已经多多少少接触过一些 ngx.re.* 的 API 了,文档也写得非常详细,我就不再一一列举了。这里,我再单独强调两个内容。

ngx.re.split

第一个是ngx.re.split。字符串切割是很常见的功能,OpenResty 也提供了对应的 API,但在社区的 QQ 交流群中,很多开发者都找不到这样的函数,只能选择自己手写。
为什么呢?其实, ngx.re.split 这个 API 并不在 lua-nginx-module 中,而是在 lua-resty-core 里面;并且它也不在 lua-resty-core 首页的文档中,而是在 lua-resty-core/lib/ngx/re.md 这个第三级目录的文档中出现的。多种原因,导致很多开发者完全不知道这个 API 的存在。
取消
完成
0/1000字
划线
笔记
复制
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
该试读文章来自付费专栏《OpenResty从入门到实战》,如需阅读全部文章,
请订阅文章所属专栏。
立即订阅
登录 后留言

精选留言(7)

  • 2xshu
    文末的问题,难道是ngx.now()取时间发生在resusme函数恢复堆栈阶段?

    作者回复: Nginx 是以性能优先作为设计理念的,它会把时间缓存下来。从 ngx.now 的源码中我们可以得到印证:
    static int
    ngx_http_lua_ngx_now(lua_State *L)
    {
        ngx_time_t *tp;

        tp = ngx_timeofday();

        lua_pushnumber(L, (lua_Number) (tp->sec + tp->msec / 1000.0L));

        return 1;
    }
    是调用了 Nginx 中的 ngx_timeofday 函数获取的时间。
    而这个函数其实是一个宏定义:
    #define ngx_timeofday() (ngx_time_t *) ngx_cached_time

    而 ngx_cached_time 的值只在函数 ngx_time_update 中会更新。
    那问题就简化为: ngx_time_update什么时候会被调用。如果你在 Nginx 的源码中去跟踪它的话,就会发现ngx_time_update的调用比较多,在事件循环中都有出现。

    2019-07-12
    4
  • helloworld
    老师,问一个困惑我的问题:
    local res, err = red:hmset("animals", t)
    if not res then
        ngx.say("failed to set animals: ", err)
        return
    end
    例如上面这种代码,其中的return有什么用呢,不加这个return不行吗

    作者回复: return 是很明确的跳出了这个函数,不再执行后面的语句。

    2019-07-14
    1
  • 姚坤
    本地Windows机器运行测试,诡异的发现ngx.worker.id是一个函数,应该写成
    if ngx.worker.id() == 0 then
        start_timer()
    end

    作者回复: windows 下这个函数是有问题的,最好在 Linux、mac 下来使用 OpenResty

    2019-09-21
  • Joshua
    这时,我们就可以在其中穿插 ngx.sleep(0),使这段代码让出控制权,让其他请求也可以得到处理。

    这里 sleep(0) 为什么 ngx.sleep(0) 会让出控制权呀

    作者回复: 因为 ngx.sleep 也是一个 yield 操作,会触发 nginx 的事件循环。

    2019-07-30
  • 回家
    看了留言以及回复,也就是说只有在调用了ngx_timer_update的时候,ngx.timer的值才会更新,而调用前者多是在事件循环中,而调用yield 函数通常是添加了一个事件。这样解释了需要yield操作之后,ngx.timer才会更新,是吧。

    作者回复: 是的,没错

    2019-07-14
  • helloworld
    我知道return的用法了,总结了一下:

    return的用法:
    用在函数中时,return 主要是用于从函数中返回结果,不会终止程序继续执行;
    用在条件语句中时,return用于终止当前程序的运行,后面的所有代码都不会执行了;
    用于循环语句时,return用于终止当前程序的运行,后面的所有代码都不会执行了,而break是终止循环继续运行,注意他们的区别;
    注意return不能直接用于代码文件级别。

    do return end一般用于调试代码的场景使用:
    用于函数中时,可放置在函数中代码的中间,这样函数剩余的代码部分不会被执行,不会中断程序执行;
    用于条件、循环语句中时,可放置在代码块的中间,会在此处中断程序执行;
    直接用于代码文件级别,会在此处中断程序执行。
    2019-07-14
  • zhang
    它只获取nginx内部的一个状态,不需要让出cpu。也就不需要lua协程和ngx core之间的切换了。



    只有期望让出cpu,让ngx corre帮他完成部分操作的api,才会出现上下文的切换,也就是yield和resume
    2019-07-12
收起评论
7
返回
顶部