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

15 | OpenResty 和别的开发平台有什么不同?

温铭 2019-06-28
你好,我是温铭。
上一模块中, 你已经学习了 OpenResty 的两个基石:NGINX 和 LuaJIT,相信你已经摩拳擦掌,准备开始学习 OpenResty 提供的 API 了吧?
不过,别着急,在这之前,你还需要再花一点儿时间,来熟悉下 OpenResty 的原理和基本概念。

原理

在前面的 LuaJIT 内容中,你已经见过下面这个架构图:
这里我再详细解释一下。
OpenResty 的 master 和 worker 进程中,都包含一个 LuaJIT VM。在同一个进程内的所有协程,都会共享这个 VM,并在这个 VM 中运行 Lua 代码。
而在同一个时间点上,每个 worker 进程只能处理一个用户的请求,也就是只有一个协程在运行。看到这里,你可能会有一个疑问:NGINX 既然能够支持 C10K (上万并发),不是需要同时处理一万个请求吗?
当然不是,NGINX 实际上是通过 epoll 的事件驱动,来减少等待和空转,才尽可能地让 CPU 资源都用于处理用户的请求。毕竟,只有单个的请求被足够快地处理完,整体才能达到高性能的目的。如果采用的是多线程模式,让一个请求对应一个线程,那么在 C10K 的情况下,资源很容易就会被耗尽的。
在 OpenResty 层面,Lua 的协程会与 NGINX 的事件机制相互配合。如果 Lua 代码中出现类似查询 MySQL 数据库这样的 I/O 操作,就会先调用 Lua 协程的 yield 把自己挂起,然后在 NGINX 中注册回调;在 I/O 操作完成(也可能是超时或者出错)后,再由 NGINX 回调 resume 来唤醒 Lua 协程。这样就完成了 Lua 协程和 NGINX 事件驱动的配合,避免在 Lua 代码中写回调。
取消
完成
0/1000字
划线
笔记
复制
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
该试读文章来自付费专栏《OpenResty从入门到实战》,如需阅读全部文章,
请订阅文章所属专栏。
立即订阅
登录 后留言

精选留言(10)

  • helloworld
    说一段比较绕口的废话来理解协程、异步:
    openresty中的一个worker进程中只有一个线程,该线程是轻量化的线程,也可以叫微线程,也可以称为协程,所谓协程就是处理协程对象的线程,所谓协程对象就是异步化的代码片段,所谓异步化就是遇到网络I/O磁盘I/O等耗时I/O操作时使CPU停止等待I/O,马上去处理下一个协程对象,避免白白浪费CPU时间,这就是非阻塞,具体说应该叫CPU非阻塞,当之前的协程对象I/O完成后,CPU在回去继续处理该协程对象剩余的代码,当然了继续处理过程中遇到I/O阻塞还会继续中断跳出去执行其他协程对象,往复...
    2019-06-28
    5
  • 阳光梦
    老师,基于worker的缓存模块,对数据改写操作时候也有竞争?
    2019-06-28
    1
    2
  • life_牛
    老师,编译openresty时候出现 error SSE 4.2 not found这个错误,网上有说cpu太老 如何处理这样的问题,还是可以忽略

    作者回复: OpenResty 的一些特性用到了 SSE 4.2 指令集,这个只能换机器了

    2019-09-12
    1
  • 宝仔
    location配置如下:
    location / {
                access_by_lua_file lua/access.lua;
                content_by_lua_file lua/content.lua;
            }

    access_by_lua_file内容如下:
    ngx.var.res = "test"
    content_by_lua_file内容如下:
    ngx.say(ngx.var.res)
    老师你好,这样子也说可以做到变量跨阶段读写的

    作者回复: 是的

    2019-09-05
  • jackstraw
    老师说的,“访问模块变量的时候,你最好保持只读,而不要尝试去修改,不然在高并发的情况下会出现 race”这个问题有点疑惑。既然每个worker同一时间只会有一个请求在进行中,那么同一时间只会有一个请求正在操作这个模块变量啊?怎么会出现race?

    作者回复: 如果修改的中途有 yield 操作,就可能会有 race。没有 yield 的话自然没有。

    2019-08-12
  • 韩健楚
    老师您好,看你的最佳实践中,遇到一个问题请教一下,在引入第三方resty库时,我根据配置请求不是返回的百度首页,是一个
    <!DOCTYPE html>
    <!--STATUS OK-->
    是什么问题?多谢。

    作者回复: 是否有更详细的信息?可以放到 gist 里面

    2019-07-02
  • FF
    温老师能否针对 OR 进程和协程的交互和他俩的协作关系展开讲讲。

    比如在同一个时间点上,每个 worker 进程只能处理一个用户的请求,也就是只有一个协程在运行。那如果协程阻塞,这个 worker 进程是否也在阻塞状态 ?协程是不是由 OR 的 worker 进程来创建 ,单个请求完后就消亡 ?

    或者这方面有什么资料可以推荐下吗?

    感谢。盼复。

    作者回复: 其实不是只有一个协程,是有一个父协程和子协程。这方面推荐 codedump 老师的源码分析:https://www.codedump.info/post/20190501-lua-stream/,内容质量很高

    2019-07-01
  • 回家
    文中这样一段话:“其余阶段都会设置一个隔离的全局变量表,以免在处理过程中污染了其他请求”。前半句似乎是要表达防止在处理的各个阶段互相干扰。后半句这样说,是因为同一个worker进程中的不同的请求其实是共享全局变量的是吧,也是同一个进程里的全局变量本来就是公用的。
    2019-06-30
  • 阳光梦
    讲的好👍
    2019-06-28
  • John
    老师,init 阶段创建的全局变量,使用content 阶段的api进行修改,是否安全?
    比如实现无reload的配置推送:
    -- 在init_by_lua 定义全局的配置文件table
    CONFIG = {a = 1};

    -- 在access_by_lua中使用全局的CONFIG实现业务逻辑
    if CONFIG['a'] = 1 then
        ngx.exit(ngx.HTTP_FORBIDDEN)
    end

    -- 为了动态跟新全局配置,提供一个API,发送get请求更新 /set-config?a =2
    location = /set-config {
          content_by_lua_block '
              local args = ngx.req.get_uri_args()
              CONFIG['a'] = args['a']
         '
    }
    2019-06-28
收起评论
10
返回
顶部