用GoFrame已经开发了一段时间,今天为大家分享避坑指南和实践干货。这些坑并不是框架设计者的问题,更多的是我作为初学者的各种摸石头过河~
Redis Setex
首先我们要知道这个概念:Redis Setex 命令为指定的 key 设置值及其过期时间。如果 key 已经存在, SETEX 命令将会替换旧的值。
在goframe中是支持这么操作的:
_, err = g.Redis().Do("SETEX", redisKey, 86400, res)
rune的使用
首先我们要知道rune是int32的别名:
type rune = int32
golang中string底层是通过byte数组实现的。中文字符在unicode下占2个字节,在utf-8编码下占3个字节,而golang默认编码正好是utf-8。
len([]rune("王中阳"))
的长度,一个中文返回一个长度,而不是返回3个字符长度。
gvalid数据校验
在goframe中,我们可以在结构体中设置校验规则,不需要在业务逻辑中进行判断,这样能使我们的代码更清晰。
示例
//校验地址 if err := gvalid.CheckStruct(ctx, data.Address, nil); err != nil { response.ParamErr(r, err) }
其中:data.Address 是需要校验的数据。在执行校验地址之前已经把传入的数据赋值到了对应的结构体:
var data *define.OrderServiceCreateReq if err := r.Parse(&data); err != nil { response.ParamErr(r, err) }
其中:define.OrderServiceCreateReq 的定义如下:
type OrderServiceCreateReq struct { SellerId uint `json:"sellerId"` AppId int `json:"appId"` ChannelId int `json:"channel_id"` OrderSn string `p:"orderSn" v:"orderSn@required#orderSn 必须传递"` Spu []*OrderServiceSkuReq `p:"spu" v:"spu@required#请输入spu"` Address *OrderServiceAddressReq `p:"address" v:"address@required#请输入address"` StrictMode bool `json:"strictMode"` PayType int `p:"payType" v:"payType@integer|in:0,1,2"` //0 余额 1 支付宝 2 微信 Rate int `json:"rate"` IsSplitMoney int `json:"is_split_money"` AccountBalance int `json:"account_balance"` ErrId int `json:"errId"` AccountPhone string `json:"account_phone"` AppKey string SiteId uint `json:"site_id"` }
其中:Address 对应的是结构体类型:OrderServiceAddressReq
我们在看一下OrderServiceAddressReq是怎么定义的?校验规则就是在这里定义的:
type OrderServiceAddressReq struct { Consignee string `v:"required|length:2,15#请输入consignee|收货人长度:min到:max位"` Phone string `v:"phone@required|phoneNew#请输入phone"` Province string `v:"required#address.province 必须传递"` City string `v:"required#address.city 必须传递"` Area string `v:"required#address.area 必须传递"` Street string Description string `v:"required|max-length:50#请输入description|详细地址最大长度不超过:max个字"` }
其中:#的后面写的就是校验提示
redis锁 避免重复请求
思路非常简单:因为redis相较于关系型数据库,读写性能更高;
这种加锁的思路简单高效:
func (s *orderApi) Create(r *ghttp.Request) { var data *define.OrderServiceCreateReq if err := r.Parse(&data); err != nil { response.ParamErr(r, err) } key := "order:" key += r.GetString("appId") key += ":" key += gmd5.MustEncryptString(r.GetBodyString()) val, err := g.Redis().DoVar("GET", key) if err != nil { response.ApiResponse(r, tools.Code{}.BackErr()) return } if val.Int() > 0 { response.ApiResponse(r, tools.Code{}.BackErrMsg("请勿重复请求")) return } _, _ = g.Redis().DoVar("SETEX", key, 3, 1) }
查文档
对了,goframe的文档做的真心不错,对开发者非常友好。
说实话,作为有几年工作经验的程序员,换一门语言进行开发难度并不大。
实现业务逻辑的思路是相通的,差别就是实现语法上,或者是语言对某些场景的支持:比如go对高并发的支持、Java对锁的支持、PHP对数组集合的支持。
我的习惯是这样:在确定了业务逻辑之后,整理好实现思路,先写伪代码,避免陷入切换语言(从PHP到Go)的不适(总是会陷入想优化代码的细枝末节里);伪代码写完后,再开始撸业务逻辑,遇到不清楚的语法直接查文档,这样做,思路就会很清晰。
比如:我要做城市信息匹配,输入参数是“北京”或者“唐山市”(直辖市不带市),但是我的数据库中存储的是“北京市”,匹配不到。
一个解决思路就是:校验一下输入参数,如果不包括“市”,就在原有的输入参数后追加一个“市”。
一查文档果然封装好了方法,拿来就用喽:
city := address.City if !gstr.Contains(city, "市") { city = city + "市" }