文章目录
- 系列文章索引
- 一、使用Json模块
- 1、引入cjson模块
- 2、table转json字符串
- 3、json字符串转table
- 4、异常处理
- (1)异常复现
- (2)使用pcall命令
- (3)cjson.safe 模块
- 5、空table返回object还是array
系列文章索引
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) end function json_decode( str ) local ok, t = pcall(_json_decode, str) if not ok then return nil end return t end local 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 then ngx.say(" --> ", type(t)) else ngx.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 --> []