Golang使用标签表达式校验结构体字段的有效性
一、背景
在服务的API接口层面,我们常常需要验证参数的有效性。 Golang中,大部分参数校验场景实际上是先将数据Bind到结构体,然后校验其字段值。
一般地,校验结构体字段值有如下两种实现方式。
- Case-By-Case 针对每个需校验的结构体字段分别写校验代码
- 优点:自由灵活,适应所有场景
- 缺点:重复且琐碎的码农工作,易使人厌烦
- 规则匹配,在结构体标签中设置预先支持的验证规则,如
email
、max:100
等形式- 优点:使用简单,不需要写琐碎的代码
- 缺点:强依赖有限的规则,缺乏灵活性,无法满足复杂场景,如多字段关联验证等
思考:有没有一种方式,即简单易用(少写代码),又能满足各种复杂的校验场景?
答案是:有!结构体标签表达式 go-tagexpr 的出现,为我们提供了兼得鱼和熊掌的第三种选择。
二、认识 go-tagexpr
go-tagexpr 允许Gopher们在 struct tag 写表达式代码,并通过高性能的解释器计算其结果。
安装
go get -u github.com/bytedance/go-tagexpr
下面使用一个小示例,演示含有枚举、比较、字段关联的较复杂场景。
示例代码
import ( "fmt" tagexpr "github.com/bytedance/go-tagexpr" ) func ExampleTagexpr() { vm := tagexpr.New("te") type Meteorology struct { Season string `te:"$=='spring'||$=='summer'||$=='autumn'||$=='winter'"` Weather string `te:"$!='snowing' || (Season)$=='winter'"` Temperature int `te:"{range:$>=-10 && $<38}{alarm:sprintf('Uncomfortable temperature: %v',$)}"` } m := &Meteorology{ Season: "summer", Weather: "snowing", Temperature: 40, } r := vm.MustRun(m) fmt.Println(r.Eval("Season")) fmt.Println(r.Eval("Weather")) fmt.Println(r.Eval("Temperature@range")) fmt.Println(r.Eval("Temperature@alarm")) // Output: // true // false // false // Uncomfortable temperature: 40 }
代码诠释:
-
新建一个标签名称为 te 的解释器
vm := tagexpr.New("te")
-
定义一个结构体,添加标签表达式,并实例化一个 m 对象。其中
$
表示当前字段值,(Season)$
表示 Season 字段的值type Meteorology struct { Season string `te:"$=='spring'||$=='summer'||$=='autumn'||$=='winter'"` Weather string `te:"$!='snowing' || (Season)$=='winter'"` Temperature int `te:"{range:$>=-10 && $<38}{alarm:sprintf('Uncomfortable temperature: %v',$)}"` } m := &Meteorology{ Season: "summer", Weather: "snowing", Temperature: 40, }
-
将对象实例 m 放入解释器中运行,返回表达式对象 r
r := vm.MustRun(m)
-
计算 Season 字段匿名表达式(
$=='spring'||$=='summer'||$=='autumn'||$=='winter'
)的值。因字段值 summer 在穷举列表中,故表达式结果为“true”r.Eval("Season")
-
计算 Weather 字段匿名表达式
$!='snowing' || (Season)$=='winter'
的值。因字段值为 snowing 且 Season 为 summer,故表达式结果为“false”r.Eval("Weather")
-
计算 Temperature 字段的
range
表达式$>=-10 && $<38
的值。因字段值为 40,超出给出的范围,所以结果为“false”r.Eval("Temperature@range")
-
计算 Temperature 字段的
alarm
表达式sprintf('Uncomfortable temperature: %v',$)
的值。这是一个调用内部函数的表达式,它打印并返回字符串,结果为“Uncomfortable temperature: 40”r.Eval("Temperature@alarm")
获取更多关于 go-expr 结构体标签表达式的语法知识 -> 查看这里
二、使用Validator校验
Validator 是有 go-expr 包提供的一个采用结构体标签表达式的参数校验组件。
主要特性
- 它要求在每个待校验字段上添加结果为布尔值的匿名表达式
- 当表达式结果为false时,表示验证不通过,此时组件将返回与该字段相关的错误信息
- 它支持使用名称为
msg
且结果为字符串的表达式作为错误信息 - 允许用户按需求自由修改错误信息的模板
- 支持各种常见的运算符
- 支持访问数组,切片,字典成员
- 支持访问当前结构体中的任何字段
- 支持访问嵌套字段,非导出字段等
- 支持注册自定义的验证函数表达式
- 内置len,sprintf,regexp,email,phone等函数表达式
安装
go get -u github.com/bytedance/go-tagexpr
我们基于前面示例稍作修改,来演示如何使用validator校验结构体字段的有效性。
示例代码
import ( "fmt" "github.com/bytedance/go-tagexpr/validator" ) func ExampleValidator() { vd := validator.New("vd") type Meteorology struct { Season string `vd:"$=='spring'||$=='summer'||$=='autumn'||$=='winter'"` Weather string `vd:"$!='snowing' || (Season)$=='winter'"` Temperature int `vd:"{@:$>=-10 && $<38}{msg:sprintf('Uncomfortable temperature: %v',$)}"` Contact string `vd:"email($)"` } m := &Meteorology{ Season: "summer", Weather: "rain", Temperature: 40, Contact: "henrylee2cn@gmail.com", } err := vd.Validate(m) if err != nil { fmt.Println(err) } // Output: // Uncomfortable temperature: 40 }
代码诠释:
-
新建一个标签名称为 vd 的校验器
vd := validator.New("vd")
-
定义一个结构体,在标签上添加校验表达式,并使用 m 实例进行测试。
type Meteorology struct { Season string `vd:"$=='spring'||$=='summer'||$=='autumn'||$=='winter'"` Weather string `vd:"$!='snowing' || (Season)$=='winter'"` Temperature int `vd:"{@:$>=-10 && $<38}{msg:sprintf('Uncomfortable temperature: %v',$)}"` Contact string `vd:"email($)"` } m := &Meteorology{ Season: "summer", Weather: "rain", Temperature: 40, Contact: "henrylee2cn@gmail.com", }
-
校验实例 m 的各字段值是否有效,如果无效,则返回error信息
err := vd.Validate(m)
注册自己的校验函数
可能你已注意到 email($)
这个表达式,它是默认注册的一个函数表达式,用于验证邮箱的有效性。其实我们也可以定义自己通用的函数表达式,以便较少标签中的代码量,增加代码复用性。
下面以 email 函数的实现为例,演示如何注册自己的校验函数:
var pattern = "^([A-Za-z0-9_\\-\\.\u4e00-\u9fa5])+\\@([A-Za-z0-9_\\-\\.])+\\.([A-Za-z]{2,8})$" emailRegexp := regexp.MustCompile(pattern) validator.RegValidateFunc("email", func(args ...interface{}) bool { if len(args) != 1 { return false } s, ok := args[0].(string) if !ok { return false } return emailRegexp.MatchString(s) }, true)
其中,validator.RegValidateFunc 的定义如下:
func RegValidateFunc(funcName string, fn func(args ...interface{}) bool, force ...bool) error
RegValidateFunc的force可选参数,表示是否强制覆盖已经注册了的同名函数。
**结论:**validator的使用方法非常简单、灵活且具有良好的扩展性,能够轻松满足各种复杂的验证场景。
获取更多关于 validator 校验器的语法知识 -> 查看这里
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
JVM内存模型——JAVA的根基
抽象 解析 程序计数器 程序计数器(Program Counter Register)是JVM中一块较小的内存区域,保存着当前线程执行的虚拟机字节码指令的内存地址(可以看作当前线程所执行的字节码的行号指示器)。 如果线程执行的是java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址(可以理解为上图所示的行号),如果正在执行的是native方法,这个计数器的值为undefined。 JVM的多线程是通过线程轮流切换并分配CPU执行时间片的方式来实现的,任何一个时刻,一个CPU都只会执行一条线程中的指令。为了保证线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各线程间的程序计数器独立存储,互不影响。 此区域是唯一一个在java虚拟机规范中没有规定任何OutOfMemoryError情况的区域,因为程序计数器是由虚拟机内部维护的,不需要开发者进行操作。 虚拟机栈 虚拟机栈(Java Virtual Machine Stacks)是线程隔离的,每创建一个线程时就会对应创建一个Java栈,即每个线程都有自己独立的虚拟机栈。这个栈中又会对应包含多个栈帧,每调用一个...
- 下一篇
RocketMQ源码分析之路由中心
微信公众号「后端进阶」,专注后端技术分享:Java、Golang、WEB框架、分布式中间件、服务治理等等。 老司机倾囊相授,带你一路进阶,来不及解释了快上车! 早期的rocketmq版本的路由功能是使用zookeeper实现的,后来rocketmq为了追求性能,自己实现了一个性能更高效且实现简单的路由中心NameServer,而且可以通过部署多个路由节点实现高可用,但它们之间并不能互相通信,这也就会导致在某一个时刻各个路由节点间的数据并不完全相同,但数据某个时刻不一致并不会导致消息发送不了,这也是rocketmq追求简单高效的一个做法。 路由启动 看了Nameserver的源码后大呼惊叹,整个NameServer总共就由这么几个类类组成: 其中NamesrvStartup为启动类,NamesrvController为核心控制器,RouteInfoManager为路由信息表。 知道了这几个类的功能之后,我们就直接定位到NamesrvStartup启动类的启动方法: org.apache.rocketmq.namesrv.NamesrvStartup#main0: public stati...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Linux系统CentOS6、CentOS7手动修改IP地址
- Docker安装Oracle12C,快速搭建Oracle学习环境
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- CentOS6,CentOS7官方镜像安装Oracle11G
- CentOS7安装Docker,走上虚拟化容器引擎之路
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- 设置Eclipse缩进为4个空格,增强代码规范
- Eclipse初始化配置,告别卡顿、闪退、编译时间过长
- 2048小游戏-低调大师作品