实现redis缓存,缓存穿透简单原理

String get(String key) {  
   String value = redis.get(key);  
   if (value  == null) {  
    if (redis.setnx(key_mutex, "1")) {  //设置分布式锁,同一时刻只能有一个线程能进行此操作
        // 3 min timeout to avoid mutex holder crash  
        redis.expire(key_mutex, 3 * 60)  
        value = db.get(key);  //为锁设置过期时间,防止持有锁的线程down了 锁没法释放,导致死锁
        redis.set(key, value);  //将过期的缓存重新放入redis
        redis.delete(key_mutex);  //操作完成后释放锁
    } else {  
        //其他线程休息50毫秒后重试  
        Thread.sleep(50);  //其他线程等待之后又重新去获取,这时候redis里面过期的数据已经被重新加载了
        get(key);  
    }  
  }  
}

func GetTopicDetail(c *gin.Context)  {
    tid:=c.Param("topic_id")
    topics:=Topics{}
    // DBHelper.Find(&topics,tid)
    //c.JSON(200,topics)
    conn:=RedisDefaultPool.Get()
    redisKey:="topic_"+tid
    defer conn.Close()
    ret,err:=redis.Bytes(conn.Do("get",redisKey))
    if err!=nil{ //缓存里没有
        DBHelper.Find(&topics,tid)
        retData,_:=ffjson.Marshal(topics)
        if topics.TopicID==0{ //代表从数据库没有匹配到
            conn.Do("setex",redisKey,20,retData) //设置20s过期时间
        }else{//正常数据 50秒缓存
            conn.Do("setex",redisKey,50,retData)
        }
        c.JSON(200,topics)
        log.Println("从数据库读取")
    }else{//代表有值
        log.Println("从 redis读取")
        ffjson.Unmarshal(ret,&topics)
        c.JSON(200,topics)
    }


}




posted @ 2019-12-20 00:34  离地最远的星  阅读(1022)  评论(0编辑  收藏  举报