GoFrame避坑指南和实践干货(2)

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: 用GoFrame已经开发了一段时间,今天为大家分享避坑指南和实践干货。这些坑并不是框架设计者的问题,更多的是我作为初学者的各种摸石头过河

用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)的不适(总是会陷入想优化代码的细枝末节里);伪代码写完后,再开始撸业务逻辑,遇到不清楚的语法直接查文档,这样做,思路就会很清晰。

比如:我要做城市信息匹配,输入参数是“北京”或者“唐山市”(直辖市不带市),但是我的数据库中存储的是“北京市”,匹配不到。

一个解决思路就是:校验一下输入参数,如果不包括“市”,就在原有的输入参数后追加一个“市”。



微信图片_20221112190254.jpg


微信图片_20221112190258.jpg


一查文档果然封装好了方法,拿来就用喽:


city := address.City
if !gstr.Contains(city, "市") {
   city = city + "市"
}


相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore     ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库 ECS 实例和一台目标数据库 RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
相关文章
|
8月前
|
安全 架构师 Java
理论实战源码齐飞!架构师社区疯传的SpringSecurity进阶小册真香
安全管理是Java应用开发中无法避免的问题,随着Spring Boot和微服务的流行,Spring Security受到越来越多Java开发者的重视,究其原因,还是沾了微服务的光。作为Spring家族中的一员,其在和Spring家族中的其他产品如SpringBoot、Spring Cloud等进行整合时,是拥有众多同类型框架无可比拟的优势的。
99 0
|
5月前
|
监控 应用服务中间件 数据库
Django 应用部署也太让人头疼了吧!但掌握这些,你就能轻松搞定,快来一探究竟!
【8月更文挑战第31天】作为一名开发者,将Django应用从开发环境顺利部署到生产环境至关重要。这涉及技术操作与实际考量,需确保应用稳定高效运行。从开发到部署至如DigitalOcean等云服务器,每一步——包括安装必要软件、配置数据库、使用Git上传代码、设置Nginx和Gunicorn、实施安全措施以及监控和日志记录——都需遵循最佳实践,以提供可靠的用户服务。
102 0
|
8月前
|
IDE 测试技术 Go
【字节跳动青训营】后端笔记整理-3 | Go语言工程实践之测试
用于验证已经修改或新增功能后,软件的既有功能是否受到影响。
139 2
|
7月前
技术经验分享:GMU简单使用一
技术经验分享:GMU简单使用一
100 0
|
7月前
|
SQL cobar 关系型数据库
程序技术好文:点我吧工作总结(技术篇)Cobar原理和环境搭建
程序技术好文:点我吧工作总结(技术篇)Cobar原理和环境搭建
56 0
|
敏捷开发 架构师 Java
GitHub上线重量级分布式架构原理设计笔记,开源的东西看着就是爽
在分布式系统中,一次业务处理可能需要多个应用来实现,比如用户发送一次下单请求,就涉及到订单系统创建订单,库存系统减库存,而对于一次下单,订单创建与减库存应该是要同时成功或者同时失效,但在分布式系统中,如果不做处理,就很有可能订单创建成功,但是减库存失败,那么解决这类问题,就需要用到分布式事务……
|
8月前
|
缓存 算法 Java
堪称神级的阿里巴巴“高并发”教程《基础+实战+源码+面试+架构》
作为一个普普通通的程序员,如何才能提升自己的能力,在职场上拥有一技之长,这也成为普通的你我,迫切的需求。
GitHub程序调优「黑马」!阿里大牛的Java性能优化实战笔记已上线
作为一个平台,JVM虚拟机起着举足轻重的作用。除了Java语言,任何一种能够被编译成字节码的计算机语言都属于Java这个平台。Groovy、Scala和JRuby等都是Java平台的一部分,它们依赖于JVM虚拟机,同时,Java平台也因为它们而变得更加丰富多彩。
|
设计模式 缓存 Java
吃透阿里2023版Java性能优化小册后,我让公司系统性能提升了200%
性能优化可以说是很多一线大厂对其公司内高级开发的基本要求(其中以Java岗最为显著)。其原因有两个:一是提高系统的性能,二是为公司节省资源。两者都能做到,那你就不可谓不是普通程序员眼中的“调优大神了”。 那么如何成为一名“调优大神”呢?
|
消息中间件 数据采集 Java
开发神技!阿里消息中间件进阶手册限时开源,请接住我的下巴
相信大家在实际工作中都用过消息中间件进行系统间数据交换,解决应用解耦、异步消息、流量削峰等问题,由此消息中间件的强大功能想必也不用我多说了!目前业界上关于消息中间件的实现多达好几十种,可谓百花齐放,所用的实现语言同样也五花八门。不管使用哪一个消息中间件,我们的目的都是实现高性能、高可用、可伸缩和最终一致性架构。