go语言后端开发学习(三)——基于validator包实现接口校验

简介: go语言后端开发学习(三)——基于validator包实现接口校验

前言

在我们开发模块的时候,有一个问题是我们必须要去考虑的,它就是如何进行入参校验,在gin框架的博客中我就介绍过一些常见的参数校验,大家可以参考gin框架学习笔记(四) ——参数绑定与参数验证,而这个其实也不是能够完全应对我们在日常开发中的需要,而今天我们要介绍的就是如何基于validator这个第三方库来实现对接口入参的校验,话不多说,开始发车!

validator包的下载

go get github.com/go-playground/validator/v10

validator的使用

背景介绍

在讲解validator包的使用方式之前我们先来看一个应用场景:博主最近在写自己的博客网站,而博客网站中有一个问题,我们先来看一下表结构:

type User struct {
  gorm.Model
  Username string `gorm:"type:varchar(20)" json:"username"`
  Password string `gorm:"type:varchar(100)" json:"password"`
  Role     int    `gorm:"type:int;default:2" json:"role"`
}

在博客中我设置了两种角色管理员(role=1)普通用户(role=2),而这也造成了一个问题:我们在添加用户按照正常业务情况是不能直接在前台创建管理员,而这个就需要我们在后端进行参数校验了(备注:其实参数校验这件事前后端都是要做的,但是我们这里介绍主要是后端,就以后端视角来介绍了),而接下来我们以给这个用户模块做参数校验为例,来介绍一下我们如何来实现参数校验以及一些其他的相关操作。

为数据模型添加校验条件

type User struct {
  gorm.Model
  Username string `gorm:"type:varchar(20)" json:"username" validate:"required,min=4,max=12"`
  Password string `gorm:"type:varchar(100)" json:"password" validate:"required,min=6 max=20"`
  Role     int    `gorm:"type:int;default:2" json:"role" validate:"required,gte=2"`
}

如上我们利用validate字段指明了数据的验证规则

拓展bindingvalidate的区别:

"validate"通常用于描述数据的验证规则,用于确保输入的数据符合特定的格式、要求或约束。验证通常在
数据提交之前进行,以确保数据的完整性和准确性。
而"binding"通常用于描述数据的绑定规则,用于将用户界面元素与数据模型进行关联。数据绑定确保用户输
入的数据与数据模型的字段正确地同步,使用户界面的更改能够反映在数据模型中,或反之亦然。因此,
"validate"主要关注数据的验证和校验,而"binding"主要关注数据在用户界面元素和数据模型之间的绑定
和同步

编写数据校验模块

我先示范一下数据消炎模块的书写,然后我再给大家具体讲解一下代码,首先我们来看一下最终我们所定义的User数据模型:

type User struct {
  gorm.Model
  Username string `gorm:"type:varchar(20)" json:"username" validate:"required,min=4,max=12" label:"用户名"`
  Password string `gorm:"type:varchar(100)" json:"password" validate:"required,min=6,max=20" label:"密码"`
  Role     int    `gorm:"type:int;default:2" json:"role" validate:"required,gte=2" label:"角色码"`
}

这里主要添加了两个标签:

  • validate:用来记录我们做入参验证的时候所做的限制条件
  • label:作为标记来返回标签值作为字段的名称

我们再来看校验模块的源代码:

package validator
import (
  "fmt"
  "gin_vue_blog/utils/errmsg"
  "github.com/go-playground/locales/zh_Hans_CN"
  ut "github.com/go-playground/universal-translator"
  "github.com/go-playground/validator/v10"
  "github.com/go-playground/validator/v10/translations/zh"
  "reflect"
)
func Validator(data any) (string, int) {
  validate := validator.New()
  uni := ut.New(zh_Hans_CN.New())
  trans, _ := uni.GetTranslator("zh_Hans_CN")
  err := zh.RegisterDefaultTranslations(validate, trans) // 注册翻译器
  if err != nil {
    fmt.Println("err:", err)
  }
  validate.RegisterTagNameFunc(func(field reflect.StructField) string {
    label := field.Tag.Get("label") // 获取label标签
    return label
  })
  err = validate.Struct(data)
  if err != nil {
    for _, err := range err.(validator.ValidationErrors) {
      return err.Translate(trans), errmsg.ERROR
    }
  }
  return "", errmsg.SUCCESS
}

我们可以来测试一下这段代码,效果如下:

我们可以看到我们成功实现了数据的校验功能,我们来看一下这一整个过程:

  1. 初始化valitor
validate := validator.New()
  1. 编写中间件,实现国际化
    validator中其默认的错误信息是英文,一定程度上会影响我们阅读,我们可以尝试将其转换为其他语言,这里我选择转换为简体中文:
uni := ut.New(zh_Hans_CN.New())
  trans, _ := uni.GetTranslator("zh_Hans_CN")
  err := zh.RegisterDefaultTranslations(validate, trans) // 注册翻译器
  if err != nil {
    fmt.Println("err:", err)
  }
  1. 进行数据校验并返回错误消息
err = validate.Struct(data)
if err != nil {
  for _, err := range err.(validator.ValidationErrors) {
    return err.Translate(trans), errmsg.ERROR
  }
}
return "", errmsg.SUCCESS
}

就这样我们就实现了一个通用的数据校验模块了。

相关文章
|
5天前
|
存储 Go API
一个go语言编码的例子
【7月更文挑战第2天】本文介绍Go语言使用Unicode字符集和UTF-8编码。Go中,`unicode/utf8`包处理编码转换,如`EncodeRune`和`DecodeRune`。`golang.org/x/text`库支持更多编码转换,如GBK到UTF-8。编码规则覆盖7位至21位的不同长度码点。
70 1
一个go语言编码的例子
|
6天前
|
安全 Go
Go语言的iota关键字有什么用途?
**Go语言中的`iota`是常量生成器,用于在`const`声明中创建递增的常量。`iota`在每个新的`const`块重置为0,然后逐行递增,简化了枚举类型或常量序列的定义。例如,定义星期枚举:** ```markdown ```go type Weekday int const ( Sunday Weekday = iota // 0 Monday // 1 Tuesday // 2 ... ) ``` 同样,`iota`可用于定义不同组的常量,如状态码和标志位,保持各自组内的递增,提高代码可读性。
|
2天前
|
监控 搜索推荐 Go
万字详解!在 Go 语言中操作 ElasticSearch
本文档通过示例代码详细介绍了如何在Go应用中使用`olivere/elastic`库,涵盖了从连接到Elasticsearch、管理索引到执行复杂查询的整个流程。
8 0
|
6天前
|
IDE Linux Go
记录一个go语言与IDE之间的问题
【7月更文挑战第1天】本文介绍在IDE中调试Go应用可能遇到的问题。当问题与IDE的自动完成有关,可以试着使用其他编辑器如Linux的vim是否无此问题。这可以验证表明IDE可能不完全兼容最新语言版本,建议使用无自动检测工具临时解决。
22 0
|
2月前
|
JSON JavaScript Go
Go 语言学习指南:变量、循环、函数、数据类型、Web 框架等全面解析
掌握 Go 语言的常见概念,如变量、循环、条件语句、函数、数据类型等等。深入了解 Go 基础知识的好起点是查阅 Go 官方文档
713 2
|
9月前
|
自然语言处理 Java Go
Go语言学习之函数
Go语言学习之函数
21 0
|
2月前
|
Java Go 数据安全/隐私保护
Go语言学习7-函数类型
本篇 Huazie 向大家介绍 Go 语言的函数类型
45 1
Go语言学习7-函数类型
|
Go API
Go学习——runtime.Caller()函数
Go学习——runtime.Caller()函数
193 0
|
Go
Go——小白学习之函数一
对于函数之前我学习的也是囫囵吞枣,这次分类型再系统的学习一遍 函数定义就不说了,我们看一下嵌套函数 此函数无返回值 有返回值的嵌套函数 如果多个函数的传参参数类型相同,返回值相同,则可以定义这个函数类型的变量。
1043 0
|
Go
Go——小白学习之函数
普通函数调用流程(调用流程:先调用后返回,先进后出) func func2(c int) { fmt.Println("c=", c) } func func1(b int) { func2(b - 1) fmt.
906 0