自定义validator和translate

本文介绍了如何在Golang中实现全局与自定义的validator和translator。Validator用于验证请求数据,当数据不符合规则时返回错误信息;Translator则用于将后端错误转化为前端可理解的语言。文中通过步骤讲解了如何配置全局验证器和翻译器,以及如何为特定数据字段创建自定义验证规则和翻译器。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

validator和translate的简介

  • validator简介:验证器是一个可以对请求数据进行验证的工具,如果不符合我们验证器定义的验证规则就返回一定的信息。
  • translator简介:翻译器是和验证器配合使用的,当我们验证不通过时,我们不能直接把后端的错误直接返回给前端页面,我们需要对错误信息进行一下翻译,让前端明白验证不通过的原因是什么,而不是直接抛出错误。

实现一个全局的validator和translator

  • 第一步我们需要一个表单数据来让我们验证,这就是我们用来绑定接收请求数据的结构体类型
type PassWordLoginForm struct {
	Mobile   string `form:"mobile" json:"mobile" binding:"required"` //先不绑定自定义validator
	//Mobile   string `form:"mobile" json:"mobile" binding:"required,mobile"` // 自定义validator(绑定mobiler的tag)
	PassWord string `form:"password" json:"password" binding:"required,min=3,max=20"`
	
}
  • 第二步我们通过valiator的engine()函数来实现配置我们的全局验证器和翻译器(这里其实gin的官方给了几个很好的例子,使用默认验证器)
import (
	"fmt"
	"github.com/gin-gonic/gin/binding"
	"github.com/go-playground/locales/en"
	"github.com/go-playground/locales/zh"
	ut "github.com/go-playground/universal-translator"
	"github.com/go-playground/validator/v10"
	//这时对包取得别名,方便理解
	en_translations "github.com/go-playground/validator/v10/translations/en"
	zh_translations "github.com/go-playground/validator/v10/translations/zh"
	"reflect"
	"strings"
)

func InitTrans(locale string) (err error) {
	//修改gin框架中的validator引擎属性, 实现定制
	if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
		//注册一个获取json的tag的自定义方法(你也可以不使用这个方法,我的表单数据是使用的)
		v.RegisterTagNameFunc(func(fld reflect.StructField) string {
			name := strings.SplitN(fld.Tag.Get("json"), ",", 2)[0]
			if name == "-" {
				return ""
			}
			//fmt.Println("______________________________________________________")
			//fmt.Println(name)
			return name
		})

		zhT := zh.New() //中文翻译器
		enT := en.New() //英文翻译器
		//第一个参数是备用的语言环境,后面的参数是应该支持的语言环境
		uni := ut.New(enT, zhT, enT)
		//需要自己定义一个全局变量ut.Translator类型的
		global.Trans, ok = uni.GetTranslator(locale)
		if !ok {
			return fmt.Errorf("uni.GetTranslator(%s)", locale)
		}

		switch locale {
		case "en":
			//根据我们初始化时传入的locale,把不同的翻译器绑定到我们验证器之中
			en_translations.RegisterDefaultTranslations(v, global.Trans)
		case "zh":
			zh_translations.RegisterDefaultTranslations(v, global.Trans)
		default:
			en_translations.RegisterDefaultTranslations(v, global.Trans)
		}
		return
	}
	return
}
  • 使用:就是把初始化放在我们gin服务启动之前就可以了,调用和官方使用默认验证器是一样的使用方式,我们只是对默认的验证器进行了一些自己的配置。
//再需要验证器的地方使用就行了
func PassWordLogin(c *gin.Context) {
	//我们需要对错误信息进行处理之后,再返回给前端(用到我们自定义的validator来实现,我们把他写在initialize包中)
	passwordLoginForm := forms.PassWordLoginForm{}
	//ShouldBind会自动判断前端请求数据时json还是xml,然后进行验证
	if err := c.ShouldBind(&passwordLoginForm); err != nil {
		//下面的错误处理逻辑,只要是表单验证我们就需要使用到
		errs, ok := err.(validator.ValidationErrors)
		if !ok {
			c.JSON(http.StatusOK, gin.H{
				"msg": err.Error(),
			})
		}
		c.JSON(http.StatusBadRequest, gin.H{
			//使用翻译器返回更友好的验证失败信息
			"error": errs.Translate(global.Trans),
		})
		return
	}
}

自定义验证器和翻译器

  • 当我们需要对某一个数据进行单独特殊的验证规则的时候,我们就需要为他绑定一个单独的验证器(同样配套一个翻译器。。。。)。
  • 第一步:同样是我们需要一个表单数据来让我们验证,这就是我们用来绑定接收请求数据的结构体类型,现在我们需要对手机号的格式进行特殊验证,如果不同过就返回失败信息。
type PassWordLoginForm struct {
	// 自定义validator(绑定mobiler的tag,记住是required后面mobile才是绑定自定义验证器的)
	Mobile   string `form:"mobile" json:"mobile" binding:"required,mobile"` 
	PassWord string `form:"password" json:"password" binding:"required,min=3,max=20"`
	
}
  • 第二步:定义我们的验证规则,官方文档中给出了格式,我就不再过多解释,详情请看gin中对于自定义验证器的文档。
//ValidateMobile 验证前端传过来的mobile数据
func ValidateMobile(fl validator.FieldLevel) bool {
	mobile := fl.Field().String()
	//使用正则表达式判断是否合法
	ok, _ := regexp.MatchString(`^1([38][0-9]|14[579]|5[^4]|16[6]|7[1-35-8]|9[189])\d{8}$`, mobile)
	if !ok {
		return false
	}
	return true
}
  • 第三步:创建我们自定义的验证器并且绑定到我们对应数据的tag(本文中的tag是mobile),自定义验证器的方式再gin官方文档的custom validator部分中有例子介绍,我就不再详细解释,同一样自定义验证器在validator的官方文档中有介绍,我们只需要使用他的translateOverride(trans ut.Translator)这个方法自定义翻译器即可。(如果你想看没有自定义翻译器的错误提示的话,可以删除掉自定义translate的部分,那个失败信息应该看的人头疼。。。。。)
//需要使用自定义的validator的话,们需要再这里绑定一个validator的engine才可以
	if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
		//把我们自定义的validator方法注册到validate(通过moblie这个tag绑定到我们对应的表单数据)
		err := v.RegisterValidation("mobile", myvalidator.ValidateMobile)
		if err != nil {
			zap.S().Fatalf("注册自定义validator失败:%s", err.Error())
			return
		}

		//自定义translate与我们的自定义validator配合使用(其实这里也需要把validator与translator进行绑定,v与global.Trans)
		v.RegisterTranslation("mobile", global.Trans, func(ut ut.Translator) error {
			//自定义失败信息
			return ut.Add("mobile", "{0} 手机号格式错误!", true) // see universal-translator for details
		}, func(ut ut.Translator, fe validator.FieldError) string {
			t, _ := ut.T("mobile", fe.Field())
			return t
		})
	}

到这自定义验证器和翻译器也就能都基本使用了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值