文章目录
系列文章索引
OpenResty使用Lua大全(一)Lua语法入门实战
OpenResty使用Lua大全(二)在OpenResty中使用Lua
OpenResty使用Lua大全(三)OpenResty使用Json模块解析json
OpenResty使用Lua大全(四)OpenResty中使用Redis
OpenResty使用Lua大全(五)OpenResty中使用MySQL
OpenResty使用Lua大全(六)OpenResty发送http请求
OpenResty使用Lua大全(七)OpenResty使用全局缓存
OpenResty使用Lua大全(八)OpenResty执行流程与阶段详解
OpenResty使用Lua大全(九)实战:nginx-lua-redis实现访问频率控制
OpenResty使用Lua大全(十)实战: Lua + Redis 实现动态封禁 IP
OpenResty使用Lua大全(十一)实战: nginx实现接口签名安全认证
OpenResty使用Lua大全(十二)实战: 动手实现一个网关框架
一、使用Json模块
web开发过程中,经常用的数据结构为json,openresty中封装了json模块。
1、引入cjson模块
local json = require(“cjson”)
json.encode 将表格(table 包含哈希键值对 和 数组键值对)数据编码为 JSON 字符串
格式:
jsonString = json.encode(表格对象)
2、table转json字符串
1)table包含哈希键值对时,数组键值将被转换为字符串键值
local json = require("cjson")
local t = {1,3,name="张三",age="19",address={"地址1","地址2"},sex="女"}
ngx.say(json.encode(t));
ngx.say("<br/>");
----{"1":1,"2":3,"sex":"女","age":"19","address":["地址1","地址2"],"name":"张三"}
local json = require("cjson")
local str = json.encode({a=1,[5]=3})
ngx.say(str); ----- {"a":1,"5":3}
ngx.say("<br/>");
2)table所有键为数组型键值对时,会当作数组看待,空位将转化为null
local json = require("cjson")
local str = json.encode({[3]=1,[5]=2,[6]="3",[7]=4})
ngx.say(str); ---- [null,null,1,null,2,"3",4]
ngx.say("<br/>");local str = json.encode({[3]=2,[5]=3})
ngx.say(str); ---- [null,null,2,null,3]
ngx.say("<br/>");
3、json字符串转table
json.decode 将JSON 字符串解码为表格对象
格式:
table = json.decode(string)
用法示例:
local json = require("cjson")
local str = [[ {"a":"v","b":2,"c":{"c1":1,"c2":2},"d":[10,11],"1":100} ]]
local t = json.decode(str)
ngx.say(" --> ", type(t))
local json = require("cjson")
local str = [[ {"a":1,"b":null} ]]
local t = json.decode(str)
ngx.say(t.a, "<br/>")
ngx.say(t.b == nil, "<br/>")
ngx.say(t.b == json.null, "<br/>")----> 1
----> false
----> true
注意:null将会转换为json.null,实际就是null
4、异常处理
(1)异常复现
local json = require("cjson")
local str = [[ {"key:"value"} ]]---少了一个双引号local t = json.decode(str)
ngx.say(" --> ", type(t))
执行请求,看看效果,执行报了–500 Internal Server Error
是因为decode方法报错了导致。我们查一下nginx的error日志:
实际情况我们希望的结果不是报错,而是返回一个友好的结果,如返回个nil
(2)使用pcall命令
如果需要在 Lua 中处理错误,必须使用函数 pcall(protected call)来包装需要执行的代码。
pcall 接收一个函数和要传递给后者的参数,并执行,执行结果:有错误、无错误;
返回值 true 或者或 false, errorinfo。
pcall 以一种"保护模式"来调用第一个参数(函数),因此 pcall 可以捕获函数执行中的任何错误。
第一个方案:重新包装一个 json decode编码
local json = require("cjson")local function _json_decode(str)return json.decode(str)
endfunction json_decode( str )local ok, t = pcall(_json_decode, str)if not ok thenreturn nilendreturn t
endlocal str = [[ {"key:"value"} ]]---少了一个双引号local t = json_decode(str)
ngx.say(t)
执行效果,没有系统错误,返回了nil
(3)cjson.safe 模块
引入cjson.safe 模块接口,该接口兼容 cjson 模块,并且在解析错误时不抛出异常,而是返回 nil。
local json = require("cjson.safe")
local str = [[ {"key:"value"} ]]local t = json.decode(str)
if t thenngx.say(" --> ", type(t))
elsengx.say("t is nil") -- 执行
end
5、空table返回object还是array
测试一下,编码空table {}
local json = require("cjson")
ngx.say("value --> ", json.encode({}))
输出 value --> {}
{}是个object;对于java的开发人员来说就不对了,空数组table,应该是[]
这个是因为对于 Lua 本身,是把数组和哈希键值对融合到一起了,所以他是无法区分空数组和空字典的。
要达到目标把 encode_empty_table_as_object
设置为 false
local json = require("cjson")
json.encode_empty_table_as_object(false)
ngx.say("value --> ", json.encode({}))
输出 value --> []