实现redis缓存装饰器,路由函数
func GetTopicDetail(context *gin.Context) {
tid:=context.Param("topic_id")
topics:=Topics{}
DBHelper.Find(&topics,tid)//从数据库取
context.Set("dbResult",topics) //这里dbResult暂时写死,应该从配置文件中读取,每一次从数据库查询到的值都会写到
}
绑定handler到路由
v1.GET("/:topic_id", CacheDecorator(GetTopicDetail,"topic_id","topic_%s",Topic{}))
实现装饰器
func CacheDecorator(h gin.HandlerFunc,param string,redKeyPattern string,empty interface{}) gin.HandlerFunc {
return func(context *gin.Context) {
//redis判断
getID:=context.Param(param) //得到ID值
redisKey:=fmt.Sprintf(redKeyPattern,getID)
conn:=RedisDefaultPool.Get()
defer conn.Close()
ret,err:=redis.Bytes(conn.Do("get",redisKey))
if err!=nil { //缓存里没有
h(context)//执行目标方法,这里执行了GetTopicDetail方法,把mysql查询到的结果放到了context中
dbResult,exists:=context.Get("dbResult") //拿出context中保存的查询结果
if !exists{
dbResult=empty //如果缓存里没有则初始化一条空数据
}
retData,_:=json.Marshal(dbResult) //序列化查询结果
conn.Do("setex",redisKey,20,retData) //存入redis
context.JSON(200,dbResult)
log.Println("从数据库读取")
}else{//缓存有 ,直接抛出
log.Println("从 redis读取")
json.Unmarshal(ret,&empty)
context.JSON(200,empty)
}
}
}