十次方项目第二天(Redis缓存和spring cache)

  为了提高查询的性能,我们通常采用Redis缓存解决。

Redis环境搭建

docker run ‐di ‐‐name=tensquare_redis ‐p 6379:6379 redis

SpringDataRedis

Spring-data-redis是spring大家族的一部分,提供了在srping应用中通过简单的配置访问
redis服务,对reids底层开发包(Jedis, JRedis, and RJC)进行了高度封装,RedisTemplate
提供了redis各种操作。

实现文章的缓存处理

查询文章操作缓存

(1)在tensquare_article 的pom.xml引入依赖

<dependency>        
 <groupId>org.springframework.boot</groupId>            
 <artifactId>spring‐boot‐starter‐data‐redis</artifactId>            
 </dependency>

(2)修改application.yml ,在spring节点下添加配置

redis:
host: 192.168.184.134

(3)修改ArticleService 引入RedisTemplate,并修改findById方法

@Autowired    
private RedisTemplate redisTemplate;    
/**    
 * 根据ID查询实体    
 * @param id    
 * @return    
 */    
public Article findById(String id) {    
//从缓存中提取        
      Article article=
      (Article)redisTemplate.opsForValue().get("article_"+id);
       
      // 如果缓存没有则到数据库查询并放入缓存        
      if(article==null) {        
      article = articleDao.findById(id).get();  
      //拿到之后要往缓存中放一份          
      redisTemplate.opsForValue().set("article_" + id, article);            
     }        
   return article;        
}

修改或删除后清除缓存

/**    
 * 修改    
 * @param article    
 */    
      public void update(Article article) {    
      redisTemplate.delete( "article_" + article.getId() );//删除缓存        
      articleDao.save(article);        
      }    
/**    
 * 删除    
 * @param id    
 */    
      public void deleteById(String id) {    
      redisTemplate.delete( "article_" + id );//删除缓存        
      articleDao.deleteById(id);        
      } 

缓存过期处理

修改findById方法 ,设置1天的过期时间

redisTemplate.opsForValue().set("article_" + id, article,1,TimeUnit.DAYS);

为了方便测试,我们可以把过期时间改为10秒

redisTemplate.opsForValue().set("article_" + id, article,10,TimeUnit.SECONDS);

SpringDataRedis其他方法

  • stringRedisTemplate.opsForValue().set("test","100",60*10,TimeUnit.SECONDS);//向redis里存入数据和设置缓存时间
  • stringRedisTemplate.opsForValue().get("test")//根据key获取缓存中的val
  • stringRedisTemplate.boundValueOps("test").increment(-1);//val做-1操作
  • stringRedisTemplate.boundValueOps("test").increment(1);//val +1
  • stringRedisTemplate.getExpire("test")//根据key获取过期时间
  • stringRedisTemplate.getExpire("test",TimeUnit.SECONDS)//根据key获取过期时间并换算成指定单位
  • stringRedisTemplate.delete("test");//根据key删除缓存
  • stringRedisTemplate.hasKey("546545");//检查key是否存在,返回boolean值
  • stringRedisTemplate.expire("red_123",1000 , TimeUnit.MILLISECONDS);//设置过期时间
  • stringRedisTemplate.opsForSet().add("red_123", "1","2","3");//向指定key中存放set集合
  • stringRedisTemplate.opsForSet().isMember("red_123", "1")//根据key查看集合中是否存在指定数据
  • stringRedisTemplate.opsForSet().members("red_123");//根据key获取set集合

Spring Cache

spring cache跟Redis没有关系,是springboot本身提供的一个缓存,无需导入Redis的包和配置文件。比Redis简单,但功能没有Redis强大,比如上面的方法一个都没有。也无法设置过期时间,所以一般用在findbyid

@Cacheable-------使用这个注解的方法在执行后会缓存其返回结果。
@CacheEvict--------使用这个注解的方法在其执行前或执行后移除Spring Cache中的某些
元素

改造上面的findbyid(使用spring cache):

(1)在GatheringService的findById方法添加缓存注解,这样当此方法第一次运行,在缓存中没有找到对应的value和key,则将查询结果放入缓存

/**    
 * 根据ID查询实体    
 * @param id    
 * @return    
 */    
/*value值随便写,代表的是缓存的名称,key就是存进去的ID,直接用#号就能拿到方法中的参数值*/
        @Cacheable(value="gathering",key="#id")    

        public Gathering findById(String id) {    

                return gatheringDao.findById(id).get();        

}

value值随便写,代表的是缓存的名称,key就是存进去的ID,直接用#号就能拿到方法中的参数值。这样写就是以后要执行findbyid就会先去名为gathering的缓存中通过ID找

(2)当我们对数据进行删改的时候,需要更新缓存。其实更新缓存也就是清除缓存,因为清除缓存后,用户再次调用查询方法无法提取缓存会重新查找数据库中的记录并放入缓存。
在GatheringService的update、deleteById方法上添加清除缓存的注解

/**    
 * 修改    
 * @param gathering    
 */    
@CacheEvict(value="gathering",key="#gathering.id") //这个地方要注意,key必须是ID
public void update(Gathering gathering) {    
    gatheringDao.save(gathering);        
}    
/**    
 * 删除    
 * @param id    
 */    
@CacheEvict(value="gathering",key="#id")    
public void deleteById(String id) {    
    gatheringDao.deleteById(id);        
} 

面试问题总结
在项目中哪部分业务用到缓存

  • findbyid都要用

你说一下项目中是如何使用缓存的
1.Redis
2.springcache
需要过期时间用Redis。不要就用springcache

说一下如何设置缓存过期时间

修改findById方法 ,设置1天的过期时间:
redisTemplate.opsForValue().set("article_" + id, article,1,TimeUnit.DAYS);
我们可以把过期时间改为10秒:
redisTemplate.opsForValue().set("article_" + id, article,10,TimeUnit.SECONDS);

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容