Golang Web开发如何做参数校验_Golang请求校验实现
技术百科
P粉602998670
发布时间:2026-01-28
浏览: 次 推荐使用 go-playground/validator 通过结构体标签统一校验请求参数,避免手写 if 判断;需封装错误格式化函数返回字段级友好提示;query 和 body 应分结构体处理,支持自定义规则且须保证可测试、无业务耦合。
用 net/http 做基础校验时,别直接 parse query/body 后手写 if 判断
Go 标准库不提供内置参数校验机制,很多人一上来就 r.ParseForm() 或 json.NewDecoder(r.Body).Decode(),再挨个检查字段是否为空、长度是否超限、数字是否越界——这样代码散落、重复多、易漏判,且无法统一错误格式。
推荐做法是:把校验逻辑收拢到结构体 + 标签中,用成熟库驱动校验流程。最轻量且可控的是 go-playground/validator,它支持 struct tag(如 validate:"required,min=1,max=50"),也兼容自定义函数和跨字段约束。
- 安装:
go get github.com/go-playground/validator/v10 - 定义结构体时带上
validatetag,比如:type CreateUserRequest struct { Name string `json:"name" validate:"required,min=2,max=20"` Age int `json:"age" validate:"required,gt=0,lt=150"` Email string `json:"email" validate:"required,email"` } - 在 handler 中统一校验:
err := validate.Struct(req),非 nil 即失败,可直接返回400 Bad Request和结构化错误信息
校验失败后怎么返回清晰的错误提示?别只 return 一个 “invalid request”
用户提交表单出错时,前端需要知道“哪个字段错了、为什么错”,而不是笼统的 400。validator 默认返回的 ValidationErrors 是个接口切片,需手动转成 map 或 slice 才好序列化。
常见处理方式是遍历错误并提取字段名和标签名(如 required、min),映射为更友好的提示:
立即学习“go语言免费学习笔记(深入)”;
- 用
err.(validator.ValidationErrors)类型断言获取原始错误 - 对每个
调用
FieldError
.Field()得字段名,.Tag()得校验规则名,.ActualTag()可得完整 tag 字符串 - 建议封装一个
formatValidationError(err error) map[string]string函数,例如:"name": "姓名长度必须在 2 到 20 个字符之间"
注意:不要在错误消息里暴露内部字段名(如 UserName),而要用 JSON tag(name)或自定义别名,否则前后端约定容易断裂。
URL query 参数和 JSON body 都要校验?别混用同一结构体无差别处理
query 和 body 的语义不同:query 天然适合简单过滤(如 ?page=1&limit=20),body 承载复杂数据(如创建资源)。虽然都能用同一个 struct,但校验粒度和规则常有差异。
- query 结构体建议加
formtag,并用decoder := schema.NewDecoder(); decoder.Decode(&q, r.URL.Query())解析(需引入go-playground/form) - body 解析仍走
json.Decode,但校验前应先检查Content-Type: application/json,避免非 JSON 请求 panic - 两者校验规则可复用,但 query 字段通常允许
omitempty,而 body 字段常要求required;例如分页参数page在 query 中可缺省,在 body 中则无意义
自定义校验规则怎么写?比如手机号、身份证号、密码强度
validator 支持注册自定义函数,用 validate.RegisterValidation 注册后,就能在 tag 里像内置规则一样使用,比如 validate:"required,phone"。
关键点在于:函数签名必须是 func(fl validator.FieldLevel) bool,其中 fl.Field().Interface() 拿到当前字段值:
- 手机号示例(简单 11 位数字):
validate.RegisterValidation("phone", func(fl validator.FieldLevel) bool { s, ok := fl.Field().Interface().(string) if !ok || len(s) != 11 { return false } for _, r := range s { if r < '0' || r > '9' { return false } } return true }) - 注册时机放在
init()或服务启动早期,确保全局唯一 - 注意性能:正则校验比纯字符串遍历慢,高频接口(如登录)建议预编译
regexp.MustCompile
真正难的不是写规则,而是让规则可测试、可复用、不耦合业务逻辑——比如密码强度校验,应只判断格式,而非查数据库或调用风控服务。
# ai
# 的是
# 放在
# 后端
# 很多人
# 是个
# 都要
# 自定义
# app
# 复用
# http
# js
# json
# go
# golang
# Error
# String
# if
# 标准库
# 字符串
# 接口
# nil
# 数据库
# git
# github
# 为什么
# red
# Interface
# 前端
# 封装
# 结构体
# Struct
# 切片
# map
# 遍历
# bool
# 中统
# 字段名
# regexp
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- 如何在Golang中定义接口_抽象方法和多态实现
- c# 在高并发场景下,委托和接口调用的性能对比
- Win10如何更改任务栏高度_Windows10解
- C++中的std::shared_from_thi
- 小程序里php怎么变mp4_小程序调用php生成m
- Python实现图数据库操作_Neo4j核心CRU
- Linux怎么实现内网穿透_Linux安装Frp客
- Linux怎么查找死循环进程_Linux系统负载分
- Win10如何备份驱动程序_Win10驱动备份步骤
- Win11怎么设置任务栏透明_Windows11使
- PythonPandas数据分析教程_数据清洗与处
- 零基础学会Python自动化办公_高效处理Exce
- Win11文件扩展名怎么显示 Win11查看文件后
- php错误怎么开启_display_errors与
- 如何在Golang中处理云原生事件_使用Event
- 如何使用Golang开发基础文件下载功能_Gola
- Win11怎么格式化U盘_Win11系统U盘格式化
- php中::能访问全局变量吗_全局作用域与类作用域
- Windows10怎么用“讲述人”读屏辅助 Win
- Win11相机打不开提示错误怎么修_相机权限开启与
- php转mp4怎么保留字幕_php处理带字幕视频转
- c++如何实现多态性_c++ 虚函数表原理与动态绑
- Win11怎么连接投影仪_Win11多显示器投屏设
- Mac的Time Machine怎么用_Mac系统
- php下载安装包太大怎么下载_分卷压缩下载方法【教
- Linux怎么禁止Root用户远程登录_Linux
- 如何在Golang中理解指针比较_Golang地址
- Windows10系统怎么查看显卡型号_Win10
- 如何在Golang中捕获结构体方法错误_Golan
- c++如何实现一个高性能的环形队列(Ring Bu
- Mac怎么进行语音输入_Mac听写功能设置与使用【
- mac怎么看硬盘大小_MAC查看磁盘存储空间与文件
- 如何使用Golang反射将map转换为struct
- c++如何判断文件是否存在_c++ filesys
- 如何使用Golang log记录不同级别日志_Go
- Windows怎样关闭锁屏广告_Windows关闭
- Win11应用商店下载慢怎么办 Win11更改DN
- Win10系统映像怎么恢复 Win10使用系统映像
- 短链接还原php提示内存不足_调整PHP内存限制设
- c++ try_emplace用法_c++ map
- Windows10系统怎么查看系统版本_Win10
- Win10电脑C盘红了怎么清理_Windows10
- Win11怎么关闭透明效果_Windows11辅助
- 如何在 PHP 单元测试中正确模拟带方法的图像处理
- c++中如何计算坐标系中两点间距离_c++勾股定理
- 如何高效识别并拦截拼接式恶意域名 spam
- Python数据挖掘核心算法实践_聚类分类与特征工
- c++输入输出流 c++ cin与cout格式化输
- Win10路由器怎么隐藏ssid Win10隐藏w
- c++怎么使用类型萃取type_traits_c+


QQ客服