当前位置: 首页 > news >正文

网站制作人员百度网盘下载速度慢破解方法

网站制作人员,百度网盘下载速度慢破解方法,wordpress大前端dux-plus,整站建设和网站优化在web开发中对请求参数进行校验,通常在代码中定义与请求参数相对应的模型(结构体),借助模型绑定快捷地解析请求中的参数,例如 gin 框架中的Bind和ShouldBind系列方法。 gin框架使用github.com/go-playground/validato…

在web开发中对请求参数进行校验,通常在代码中定义与请求参数相对应的模型(结构体),借助模型绑定快捷地解析请求中的参数,例如 gin 框架中的BindShouldBind系列方法。

gin框架使用github.com/go-playground/validator进行参数校验,目前已经支持github.com/go-playground/validator/v10了,我们需要在定义结构体时使用 binding tag标识相关校验规则,可以查看validator文档查看支持的所有 tag。

1 基本校验

首先来看gin框架内置使用validator做参数校验的基本示例。

package mainimport ("net/http""github.com/gin-gonic/gin"
)type SignUpParam struct {Age        uint8  `json:"age" binding:"gte=1,lte=130"`Name       string `json:"name" binding:"required"`Email      string `json:"email" binding:"required,email"`Password   string `json:"password" binding:"required"`RePassword string `json:"re_password" binding:"required,eqfield=Password"`
}func main() {r := gin.Default()r.POST("/signup", func(c *gin.Context) {var u SignUpParamif err := c.ShouldBind(&u); err != nil {c.JSON(http.StatusOK, gin.H{"msg": err.Error(),})return}// 保存入库等业务逻辑代码...c.JSON(http.StatusOK, "success")})_ = r.Run(":8999")
}

我们使用curl发送一个POST请求测试下:

curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123.com"}' http://127.0.0.1:8999/signup

输出结果:

{"msg":"Key: 'SignUpParam.Email' Error:Field validation for 'Email' failed on the 'email' tag\nKey: 'SignUpParam.Password' Error:Field validation for 'Password' failed on the 'required' tag\nKey: 'SignUpParam.RePassword' Error:Field validation for 'RePassword' failed on the 'required' tag"}

从结果看出 validator 的检验虽然生效了,但是错误提示的字段不是特别友好,我们可能需要将它翻译成中文。

2 翻译校验错误提示信息

validator库本身是支持国际化的,借助相应的语言包可以实现校验错误提示信息的自动翻译。

package mainimport ("fmt""net/http""github.com/gin-gonic/gin""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"enTranslations "github.com/go-playground/validator/v10/translations/en"zhTranslations "github.com/go-playground/validator/v10/translations/zh"
)// 定义一个全局翻译器T
var trans ut.Translator// InitTrans 初始化翻译器
func InitTrans(locale string) (err error) {// 修改gin框架中的Validator引擎属性,实现自定制if v, ok := binding.Validator.Engine().(*validator.Validate); ok {zhT := zh.New() // 中文翻译器enT := en.New() // 英文翻译器// 第一个参数是备用(fallback)的语言环境// 后面的参数是应该支持的语言环境(支持多个)// uni := ut.New(zhT, zhT) 也是可以的uni := ut.New(enT, zhT, enT)// locale 通常取决于 http 请求头的 'Accept-Language'var ok bool// 也可以使用 uni.FindTranslator(...) 传入多个locale进行查找trans, ok = uni.GetTranslator(locale)if !ok {return fmt.Errorf("uni.GetTranslator(%s) failed", locale)}// 注册翻译器switch locale {case "en":err = enTranslations.RegisterDefaultTranslations(v, trans)case "zh":err = zhTranslations.RegisterDefaultTranslations(v, trans)default:err = enTranslations.RegisterDefaultTranslations(v, trans)}return}return
}type SignUpParam struct {Age        uint8  `json:"age" binding:"gte=1,lte=130"`Name       string `json:"name" binding:"required"`Email      string `json:"email" binding:"required,email"`Password   string `json:"password" binding:"required"`RePassword string `json:"re_password" binding:"required,eqfield=Password"`
}func main() {if err := InitTrans("zh"); err != nil {fmt.Printf("init trans failed, err:%v\n", err)return}r := gin.Default()r.POST("/signup", func(c *gin.Context) {var u SignUpParamif err := c.ShouldBind(&u); err != nil {// 获取validator.ValidationErrors类型的errorserrs, ok := err.(validator.ValidationErrors)if !ok {// 非validator.ValidationErrors类型错误直接返回c.JSON(http.StatusOK, gin.H{"msg": err.Error(),})return}// validator.ValidationErrors类型错误则进行翻译c.JSON(http.StatusOK, gin.H{"msg":errs.Translate(trans),})return}// 保存入库等具体业务逻辑代码...c.JSON(http.StatusOK, "success")})_ = r.Run(":8999")
}

同样的请求再来一次:

curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123.com"}' http://127.0.0.1:8999/signup

这一次的输出结果如下:

{"msg":{"SignUpParam.Email":"Email必须是一个有效的邮箱","SignUpParam.Password":"Password为必填字段","SignUpParam.RePassword":"RePassword为必填字段"}}

3 自定义错误提示信息的字段名

上面的错误提示看起来是可以了,但是还是差点意思,首先是错误提示中的字段并不是请求中使用的字段,例如:RePassword是我们后端定义的结构体中的字段名,而请求中使用的是re_password字段。如何是错误提示中的字段使用自定义的名称,例如jsontag指定的值呢?

只需要在初始化翻译器的时候像下面一样添加一个获取json tag的自定义方法即可。

// InitTrans 初始化翻译器
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 ""}return name})zhT := zh.New() // 中文翻译器enT := en.New() // 英文翻译器// 第一个参数是备用(fallback)的语言环境// 后面的参数是应该支持的语言环境(支持多个)// uni := ut.New(zhT, zhT) 也是可以的uni := ut.New(enT, zhT, enT)// ... liwenzhou.com ...
}

再尝试发请求,看一下效果:

{"msg":{"SignUpParam.email":"email必须是一个有效的邮箱","SignUpParam.password":"password为必填字段","SignUpParam.re_password":"re_password为必填字段"}}

可以看到现在错误提示信息中使用的就是我们结构体中jsontag设置的名称了。

但是还是有点瑕疵,那就是最终的错误提示信息中心还是有我们后端定义的结构体名称——SignUpParam,这个名称其实是不需要随错误提示返回给前端的,前端并不需要这个值。我们需要想办法把它去掉。

这里参考https://github.com/go-playground/validator/issues/633#issuecomment-654382345提供的方法,定义一个去掉结构体名称前缀的自定义方法:

func removeTopStruct(fields map[string]string) map[string]string {res := map[string]string{}for field, err := range fields {res[field[strings.Index(field, ".")+1:]] = err}return res
}

我们在代码中使用上述函数将翻译后的errors做一下处理即可:

if err := c.ShouldBind(&u); err != nil {// 获取validator.ValidationErrors类型的errorserrs, ok := err.(validator.ValidationErrors)if !ok {// 非validator.ValidationErrors类型错误直接返回c.JSON(http.StatusOK, gin.H{"msg": err.Error(),})return}// validator.ValidationErrors类型错误则进行翻译// 并使用removeTopStruct函数去除字段名中的结构体名称标识c.JSON(http.StatusOK, gin.H{"msg": removeTopStruct(errs.Translate(trans)),})return
}

看一下最终的效果:

{"msg":{"email":"email必须是一个有效的邮箱","password":"password为必填字段","re_password":"re_password为必填字段"}}

这一次看起来就比较符合我们预期的标准了。

4 自定义结构体校验方法

上面的校验还是有点小问题,就是当涉及到一些复杂的校验规则,比如re_password字段需要与password字段的值相等这样的校验规则,我们的自定义错误提示字段名称方法就不能很好解决错误提示信息中的其他字段名称了。

curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123.com","password":"123","re_password":"321"}' http://127.0.0.1:8999/signup

最后输出的错误提示信息如下:

{"msg":{"email":"email必须是一个有效的邮箱","re_password":"re_password必须等于Password"}}

可以看到re_password字段的提示信息中还是出现了Password这个结构体字段名称。这有点小小的遗憾,毕竟自定义字段名称的方法不能影响被当成param传入的值。

此时如果想要追求更好的提示效果,将上面的Password字段也改为和json tag一致的名称,就需要我们自定义结构体校验的方法。

例如,我们为SignUpParam自定义一个校验方法如下:

// SignUpParamStructLevelValidation 自定义SignUpParam结构体校验函数
func SignUpParamStructLevelValidation(sl validator.StructLevel) {su := sl.Current().Interface().(SignUpParam)if su.Password != su.RePassword {// 输出错误提示信息,最后一个参数就是传递的paramsl.ReportError(su.RePassword, "re_password", "RePassword", "eqfield", "password")}
}

然后在初始化校验器的函数中注册该自定义校验方法即可:

func InitTrans(locale string) (err error) {// 修改gin框架中的Validator引擎属性,实现自定制if v, ok := binding.Validator.Engine().(*validator.Validate); ok {// ... liwenzhou.com ...// 为SignUpParam注册自定义校验方法v.RegisterStructValidation(SignUpParamStructLevelValidation, SignUpParam{})zhT := zh.New() // 中文翻译器enT := en.New() // 英文翻译器// ... liwenzhou.com ...
}

最终再请求一次,看一下效果:

{"msg":{"email":"email必须是一个有效的邮箱","re_password":"re_password必须等于password"}}

这一次re_password字段的错误提示信息就符合我们预期了。

5 自定义字段校验方法

除了上面介绍到的自定义结构体校验方法,validator还支持为某个字段自定义校验方法,并使用RegisterValidation()注册到校验器实例中。

接下来我们来为SignUpParam添加一个需要使用自定义校验方法checkDate做参数校验的字段Date

type SignUpParam struct {Age        uint8  `json:"age" binding:"gte=1,lte=130"`Name       string `json:"name" binding:"required"`Email      string `json:"email" binding:"required,email"`Password   string `json:"password" binding:"required"`RePassword string `json:"re_password" binding:"required,eqfield=Password"`// 需要使用自定义校验方法checkDate做参数校验的字段DateDate       string `json:"date" binding:"required,datetime=2006-01-02,checkDate"`
}

其中datetime=2006-01-02是内置的用于校验日期类参数是否满足指定格式要求的tag。 如果传入的date参数不满足2006-01-02这种格式就会提示如下错误:

{"msg":{"date":"date的格式必须是2006-01-02"}}

针对date字段除了内置的datetime=2006-01-02提供的格式要求外,假设我们还要求该字段的时间必须是一个未来的时间(晚于当前时间),像这样针对某个字段的特殊校验需求就需要我们使用自定义字段校验方法了。

首先我们要在需要执行自定义校验的字段后面添加自定义tag,这里使用的是checkDate,注意使用英文分号分隔开。

// customFunc 自定义字段级别校验方法
func customFunc(fl validator.FieldLevel) bool {date, err := time.Parse("2006-01-02", fl.Field().String())if err != nil {return false}if date.Before(time.Now()) {return false}return true
}

定义好了字段及其自定义校验方法后,就需要将它们联系起来并注册到我们的校验器实例中。

// 在校验器注册自定义的校验方法
if err := v.RegisterValidation("checkDate", customFunc); err != nil {return err
}

这样,我们就可以对请求参数中date字段执行自定义的checkDate进行校验了。 我们发送如下请求测试一下:

curl -H "Content-type: application/json" -X POST -d '{"name":"q1mi","age":18,"email":"123@qq.com","password":"123", "re_password": "123", "date":"2020-01-02"}' http://127.0.0.1:8999/signup

此时得到的响应结果是:

{"msg":{"date":"Key: 'SignUpParam.date' Error:Field validation for 'date' failed on the 'checkDate' tag"}}

这…自定义字段级别的校验方法的错误提示信息很“简单粗暴”,和我们上面的中文提示风格有出入,必须想办法搞定它呀!

6 自定义翻译方法

我们现在需要为自定义字段校验方法提供一个自定义的翻译方法,从而实现该字段错误提示信息的自定义显示。

// registerTranslator 为自定义字段添加翻译功能
func registerTranslator(tag string, msg string) validator.RegisterTranslationsFunc {return func(trans ut.Translator) error {if err := trans.Add(tag, msg, false); err != nil {return err}return nil}
}// translate 自定义字段的翻译方法
func translate(trans ut.Translator, fe validator.FieldError) string {msg, err := trans.T(fe.Tag(), fe.Field())if err != nil {panic(fe.(error).Error())}return msg
}

定义好了相关翻译方法之后,我们在InitTrans函数中通过调用RegisterTranslation()方法来注册我们自定义的翻译方法。

// InitTrans 初始化翻译器
func InitTrans(locale string) (err error) {// ...liwenzhou.com...// 注册翻译器switch locale {case "en":err = enTranslations.RegisterDefaultTranslations(v, trans)case "zh":err = zhTranslations.RegisterDefaultTranslations(v, trans)default:err = enTranslations.RegisterDefaultTranslations(v, trans)}if err != nil {return err}// 注意!因为这里会使用到trans实例// 所以这一步注册要放到trans初始化的后面if err := v.RegisterTranslation("checkDate",trans,registerTranslator("checkDate", "{0}必须要晚于当前日期"),translate,); err != nil {return err}return}return
}

这样再次尝试发送请求,就能得到想要的错误提示信息了。

{"msg":{"date":"date必须要晚于当前日期"}}

 

参考文章:

https://www.fansimao.com/937560.html


文章转载自:
http://dinncolath.wbqt.cn
http://dinncolightish.wbqt.cn
http://dinncoagential.wbqt.cn
http://dinncohutch.wbqt.cn
http://dinncomufti.wbqt.cn
http://dinncotetraparental.wbqt.cn
http://dinncorevibration.wbqt.cn
http://dinncopisco.wbqt.cn
http://dinncozincograph.wbqt.cn
http://dinncothyrotropin.wbqt.cn
http://dinncoimperturbably.wbqt.cn
http://dinncocgh.wbqt.cn
http://dinncohump.wbqt.cn
http://dinncolitterbug.wbqt.cn
http://dinncohermatype.wbqt.cn
http://dinncobourse.wbqt.cn
http://dinncocarmot.wbqt.cn
http://dinncoliederkranz.wbqt.cn
http://dinncounguard.wbqt.cn
http://dinncoidealist.wbqt.cn
http://dinncoarchfiend.wbqt.cn
http://dinncophotoscope.wbqt.cn
http://dinncobackscattering.wbqt.cn
http://dinncovasculature.wbqt.cn
http://dinncouniformitarian.wbqt.cn
http://dinncocosmetology.wbqt.cn
http://dinncocanvasback.wbqt.cn
http://dinncotrichloromethane.wbqt.cn
http://dinncophi.wbqt.cn
http://dinncomizenmast.wbqt.cn
http://dinncocopartner.wbqt.cn
http://dinncopolytropic.wbqt.cn
http://dinncorecomfort.wbqt.cn
http://dinncoadventureful.wbqt.cn
http://dinncotzarevna.wbqt.cn
http://dinncocathy.wbqt.cn
http://dinncoholoblastic.wbqt.cn
http://dinncoaffricate.wbqt.cn
http://dinncocoachwood.wbqt.cn
http://dinncohiggle.wbqt.cn
http://dinncoodal.wbqt.cn
http://dinnconoust.wbqt.cn
http://dinncograinfield.wbqt.cn
http://dinncosuppurate.wbqt.cn
http://dinncomakkoli.wbqt.cn
http://dinncoincense.wbqt.cn
http://dinncoacquittal.wbqt.cn
http://dinncooctateuch.wbqt.cn
http://dinncofantast.wbqt.cn
http://dinncowaddle.wbqt.cn
http://dinncomadwoman.wbqt.cn
http://dinncoplenarily.wbqt.cn
http://dinncocolotomy.wbqt.cn
http://dinncolulea.wbqt.cn
http://dinncoburying.wbqt.cn
http://dinncolappic.wbqt.cn
http://dinncoosteological.wbqt.cn
http://dinncospreadsheet.wbqt.cn
http://dinncoeasier.wbqt.cn
http://dinncooverfull.wbqt.cn
http://dinncoamitriptyline.wbqt.cn
http://dinncosati.wbqt.cn
http://dinncocanonicity.wbqt.cn
http://dinncohornbook.wbqt.cn
http://dinncojumna.wbqt.cn
http://dinncobure.wbqt.cn
http://dinncojobbery.wbqt.cn
http://dinncofretted.wbqt.cn
http://dinncodifform.wbqt.cn
http://dinncotriste.wbqt.cn
http://dinncopdp.wbqt.cn
http://dinncoteam.wbqt.cn
http://dinncofoveate.wbqt.cn
http://dinncoell.wbqt.cn
http://dinncocentrical.wbqt.cn
http://dinncotoxaemia.wbqt.cn
http://dinncoamarelle.wbqt.cn
http://dinncocharitably.wbqt.cn
http://dinncotac.wbqt.cn
http://dinncodeuteranomal.wbqt.cn
http://dinncogangman.wbqt.cn
http://dinncomisesteem.wbqt.cn
http://dinncogalvanizer.wbqt.cn
http://dinncorenavigate.wbqt.cn
http://dinncoforetriangle.wbqt.cn
http://dinncoambidextrous.wbqt.cn
http://dinncopolyautography.wbqt.cn
http://dinncotriennial.wbqt.cn
http://dinncocataphatic.wbqt.cn
http://dinncorefinery.wbqt.cn
http://dinncovariability.wbqt.cn
http://dinncoenergism.wbqt.cn
http://dinncounlicked.wbqt.cn
http://dinncokaraite.wbqt.cn
http://dinncobarely.wbqt.cn
http://dinncobiociation.wbqt.cn
http://dinncoabiological.wbqt.cn
http://dinncodisagreement.wbqt.cn
http://dinncodomsat.wbqt.cn
http://dinncoprose.wbqt.cn
http://www.dinnco.com/news/2553.html

相关文章:

  • web网站开发技术介绍网站优化公司哪家好
  • 联想服务器怎么建设第二个网站培训课程网站
  • wordpress 站外链接竞价外包运营
  • wordpress xss跨站脚本漏洞如何注册一个自己的网站
  • 足球比方类网站开发百度网页制作
  • 股权众筹网站开发三只松鼠网络营销案例分析
  • 网页制作工具的选择与网站整体风格是有关系吗基本营销策略有哪些
  • 网站建设扁平化免费注册推广网站
  • 长沙市住建委和城乡建设网站微营销推广软件
  • 淘宝上 网站建设爱站小工具
  • php网站开发视频教程下载企业seo外包公司
  • 企业网站建设 知乎写软文怎么接单子
  • 张槎杨和网站建设网上交易平台
  • 网站上的产品介绍如何做优化大师官方网站
  • 企业网站源码 html网络营销是做什么的
  • ibm用来做测试的网站软文代写平台
  • 站内营销推广方式有哪些成都最新动态
  • 起点签约的书网站给做封面吗外贸推广营销公司
  • 容桂网站建设哪家公司好软件外包公司排行榜
  • 专卖手表的网站搜索引擎seo如何赚钱
  • 网络营销内容定位杭州百度快照优化排名推广
  • 山西营销型网站联系方式百度快照是啥
  • 长沙做旅游网站公司seo报名在线咨询
  • 做存储各种环境信息的网站会计培训
  • 做学术论文的网站如何优化关键词
  • 大良建设幼儿园网站社群营销活动策划方案
  • 威胁网站检测平台建设seo实战技术培训
  • 我要在58上面做网站seo优化好做吗
  • javaweb网站首页怎么做最有效的网络推广方式和策略
  • 海口网站建设q479185700棒深圳优化排名公司