新闻动态

go面向对象方式操作JSON库实现四则运算

发布日期:2022-07-15 19:40 | 文章来源:站长之家

在之前实现的JSON解析器中当时只实现了将一个 JSON 字符串转换为一个JSONObject,并没有将其映射为一个具体的struct;如果想要获取值就需要先做断言将其转换为map或者是切片再来获,会比较麻烦。

decode, err := xjson.Decode(`{"glossary":{"title":"example glossary","age":1}}`)
assert.Nil(t, err)
glossary := v["glossary"].(map[string]interface{})
assert.Equal(t, glossary["title"], "example glossary")
assert.Equal(t, glossary["age"], 1)

但其实转念一想,部分场景我们甚至我们只需要拿到JSON中的某个字段的值,这样还需要先声明一个struct会略显麻烦。

于是我也打算增加类似的功能,使用方式如下:

最后还加上了一个四则运算的功能。

面向对象的方式操作 JSON

因为功能类似,所以我参考了tidwallAPI但去掉一些我觉得暂时用不上的特性,并调整了一点语法。

当前这个版本只能通过确定的key加上.点符号访问数据,如果是数组则用[index]的方式访问下标。
[]符号访问数组我觉得要更符合直觉一些。

以下是一个包含多重嵌套JSON的访问示例:

str := `
{
"name": "bob",
"age": 20,
"skill": {
 "lang": [
  {
"go": {
 "feature": [
  "goroutine",
  "channel",
  "simple",
  true
 ]
}
  }
 ]
}
}`
name := xjson.Get(str, "name")
assert.Equal(t, name.String(), "bob")
age := xjson.Get(str, "age")
assert.Equal(t, age.Int(), 20)
assert.Equal(t, xjson.Get(str,"skill.lang[0].go.feature[0]").String(), "goroutine")
assert.Equal(t, xjson.Get(str,"skill.lang[0].go.feature[1]").String(), "channel")
assert.Equal(t, xjson.Get(str,"skill.lang[0].go.feature[2]").String(), "simple")
assert.Equal(t, xjson.Get(str,"skill.lang[0].go.feature[3]").Bool(), true)

这样的语法使用个人觉得还是满符合直觉的,相信对使用者来说也比较简单。

返回值参考了tidwall使用了一个Result对象,它提供了多种方法可以方便的获取各种类型的数据

func (r Result) String() string
func (r Result) Bool() bool
func (r Result) Int() int
func (r Result) Float() float64
func (r Result) Map() map[string]interface{}
func (r Result) Array() *[]interface{}
func (r Result) Exists() bool

比如使用Map()/Array()这两个函数可以将JSON数据映射到map和切片中,当然前提是传入的语法返回的是一个合法JSONObject或数组。

实现原理

在实现之前需要先定义一个基本语法,主要支持以下四种用法:

  • 单个key的查询:Get(json,"name")
  • 嵌套查询:Get(json,"obj1.obj2.obj3.name")
  • 数组查询:Get(json,"obj.array[0]")
  • 数组嵌套查询:Get(json,"obj.array[0].obj2.obj3[1].name")

语法很简单,符合我们日常接触到语法规则,这样便可以访问到JSON数据中的任何一个值。

其实实现过程也不复杂,我们已经在上一文中实现将JSON字符串转换为一个JSONObject了。

这次只是额外再解析刚才定义的语法为token,然后解析该token的同时再从生成好的JSONObject中获取数据。

最后在解析完token时拿到的JSONObject数据返回即可。

我们以这段查询代码为例:

首先第一步是对查询语法做词法分析,最终得到下图的token

在词法分析过程中也可以做简单的语法校验;比如如果包含数组查询,并不是以]符号结尾时就抛出语法错误。

接着我们遍历语法的 token。如下图所示:

每当遍历到token类型为Key时便从当前的 JSONObject 对象中获取数据,并用获取到的值替覆盖为当前的 JSONObject。

其中每当遇到.[]这样的 token 时便消耗掉,直到我们将 token 遍历完毕,这时将当前JSONObject返回即可。

在遍历过程中当遇到非法格式时,比如obj_list[1.]便会返回一个空的JSONObject

语法校验这点其实也很容易办到,因为根据我们的语法规则,Array中的index后一定紧接的是一个EndArray,只要不是一个EndArray便能知道语法不合法了。

有兴趣的可以看下解析过程的源码:

https://github.com/crossoverJie/xjson/blob/cfbca51cc9bc0c77e6cb9c9ad3f964b2054b3826/json.go#L46

对 JSON 做四则运算

str := `{"name":"bob", "age":10,"magic":10.1, "score":{"math":[1,2]}}`
result := GetWithArithmetic(str, "(age+age)*age+magic")
assert.Equal(t, result.Float(), 210.1)
result = GetWithArithmetic(str, "(age+age)*age")
assert.Equal(t, result.Int(), 200)
result = GetWithArithmetic(str, "(age+age) * age + score.math[0]")
assert.Equal(t, result.Int(), 201)
result = GetWithArithmetic(str, "(age+age) * age - score.math[0]")
assert.Equal(t, result.Int(), 199)
result = GetWithArithmetic(str, "score.math[1] / score.math[0]")
assert.Equal(t, result.Int(), 2)

最后我还扩展了一下语法,可以支持对JSON数据中的整形(int、float)做四则运算,虽然这是一个小众需求,但做完我觉得还挺有意思的,目前在市面上我还没发现有类似功能的库,可能和小众需求有关

版权声明:本站文章来源标注为YINGSOO的内容版权均为本站所有,欢迎引用、转载,请保持原文完整并注明来源及原文链接。禁止复制或仿造本网站,禁止在非www.yingsoo.com所属的服务器上建立镜像,否则将依法追究法律责任。本站部分内容来源于网友推荐、互联网收集整理而来,仅供学习参考,不代表本站立场,如有内容涉嫌侵权,请联系alex-e#qq.com处理。

相关文章

实时开通

自选配置、实时开通

免备案

全球线路精选!

全天候客户服务

7x24全年不间断在线

专属顾问服务

1对1客户咨询顾问

在线
客服

在线客服:7*24小时在线

客服
热线

400-630-3752
7*24小时客服服务热线

关注
微信

关注官方微信
顶部