前言

近期会更新一系列开源项目的文章,新的一年会和大家做更多的开源项目,也欢迎大家加入进来。

xutil

今天分享的文章源自于开源项目jinzaigo/xutil的封装。

在封装过程中,劲仔将实现原理以及相关实践思考,写成文章分享出来,从而汇总系列文章集合。

PHP转Go

我和劲仔都是PHP转Go,身边越来越多做PHP的朋友也逐渐在用Go进行重构,重构过程中,会发现php的json解析操作(系列化与反序列化)是真的香,弱类型语言的各种隐式类型转换,很大程度的减低了程序的复杂度。

反观go使用标准库encoding/json,来做json解析就没有那么愉快了(只要数据类型定义不对,就很容易抛error)

JSON解析实践

案例:用go重构的服务,对接的上游还是php服务,这时php接口输出的json串为{"name":"AppleWatchS8","price":"3199"}

其中price字段应该得为float类型,但由于php弱类型语言,没有强制约束输出类型的机制,就很容易出现这种输出类型不对的情况,然后到go服务里得怎么处理呢?

标准库encoding/json

package main

import (
"encoding/json"
"fmt"
) type ProductInfo struct {
Name string `json:"name"`
Price float32 `json:"price"`
} func main() {
str := "{"name":"AppleWatchS8","price":"3199"}"
data := ProductInfo{}
if err := json.Unmarshal([]byte(str), &data); err != nil {
fmt.Println("error: " + err.Error())
} else {
fmt.Println(data)
}
} //输出结果
//error: json: cannot unmarshal string into Go struct field ProductInfo.price of type float32

显然,使用go标准库做json解析,是应对不了这种类型不一致的情况的。下面则借助第三方库的能力来做处理

第三方库json-iterator

简单介绍:

执行速度:jsoniter 的 Golang 版本可以比标准库(encoding/json)快 6 倍之多

两个特点:

  1. 完全兼容标准库,也就是API用法完全一样,原有逻辑代码不需要改动,只需要替换import包名

  2. 提供了一个PHP兼容模式,可以自动转换字符串/数字弱类型问题,可以将空数组[]转换为空结构体(解决PHP中的array输出为[]的问题)。注意,该兼容模式需要手动开启

安装方式:

go get -u github.com/json-iterator/go

具体代码实现:

package main

import (
"fmt"
jsoniter "github.com/json-iterator/go"
"github.com/json-iterator/go/extra"
) var json = jsoniter.ConfigCompatibleWithStandardLibrary func init() {
extra.RegisterFuzzyDecoders() //开启PHP兼容模式
} type ProductInfo struct {
Name string `json:"name"`
Price float32 `json:"price"`
} func main() {
str := "{"name":"AppleWatchS8","price":"3199"}"
data := ProductInfo{}
if err := json.Unmarshal([]byte(str), &data); err != nil {
fmt.Println("error: " + err.Error())
} else {
fmt.Println(data)
}
} //输出结果
//{AppleWatchS8 3199}

看输出结果,会发现用了这个库并且开启了PHP兼容模式,json中price字段string类型,就会自动转换为结构体中定义的float32类型。

这样我们在使用price字段处理业务逻辑时,就只需要按float32做处理就行,不用进行类型断言。这个库解决了json解析类型转换问题的同时,也能极大的提高我们开发效率。

收集到开源工具包xutil中

这个第三库用起来如此方便,那肯定是要收录进来的,将替换包名、手动开启PHP兼容模式、还有常用的API方法(系列化与反序列化操作),统一封装进来,简化使用流程。

同时,为了便于后续扩展更多的兼容模式,所以将代码都放在xjson目录下

以上这个思路也适用于大家封装自己内部使用的工具库。

使用示例:

go get -u github.com/jinzaigo/xutil之后,

import github.com/jinzaigo/xutil/xjson

即可通过xjson.Unmarshal()等方法,进行json解析操作。

示例代码:

package main

import (
"fmt"
"github.com/jinzaigo/xutil/xjson"
) type ProductInfo struct {
Name string `json:"name"`
Price float32 `json:"price"`
} func main() {
str := "{"name":"AppleWatchS8","price":"3199"}"
data := ProductInfo{}
if err := xjson.Unmarshal([]byte(str), &data); err != nil {
fmt.Println("error: " + err.Error())
} else {
fmt.Println(data)
}
}

总结

业务系统从php转go,或go对接php服务,都会遇到这个因为数据类型不一致导致json解析错误的共性问题。

使用第三方库json-iterator能很好的解决我们的痛点,并且比标准库执行速度还更快。

收录到开源项目中,更好的帮助到需要的朋友,欢迎使用、star与PR共同建设。

https://github.com/jinzaigo/xutil

最新文章

  1. The note of Vue.js
  2. 【代码笔记】iOS-替换电话号码中间4位为-号
  3. bzoj3998: [TJOI2015]弦论
  4. Colorable Fantasy UI
  5. 【MySQL】数据行长度的一些限制
  6. 九度OJ 1544 数字序列区间最小值
  7. 学习PHP时的一些总结(二)
  8. Objective-c 程序结构
  9. BeautifulSoup抓取列表页锚文本
  10. 引入的ajax中异步添加联系人
  11. Java中实现多线程关键词整理
  12. Web安全基础——小白自学
  13. MongoDB的ORM框架——Morphia
  14. CentOS7下安装Python3并保留Python2
  15. Centos6安装和配置etcd3
  16. Python 捕捉traceback异常栈信息
  17. flask get和post请求使用
  18. Tesseract-OCR的简单使用与训练
  19. 【Big Data - Hadoop - MapReduce】初学Hadoop之图解MapReduce与WordCount示例分析
  20. MySQL中tinytext、text、mediumtext和longtext详解【转】

热门文章

  1. 【神经网络】softmax回归
  2. 第三方代开的微信小程序更换管理员
  3. 浅入浅出 1.7和1.8的 HashMap
  4. Java Timer使用介绍
  5. P6492 STEP(线段树维护左右区间pushup)
  6. 本人常用的sed命令用法
  7. 设计模式学习(二十四):Spring 中使用到的设计模式
  8. 一、Redis的Java客户端
  9. PGL图学习之图神经网络GraphSAGE、GIN图采样算法[系列七]
  10. bugku 秋名山老司机