java redis缓存框架cacheChen

前世今生

java开发中,针对主流网站,流量大的网站都会使用缓存,目前流行的缓存框架很多,其中表现的最优秀的redis,算是佼佼者。

cacheChen 是什么

框架基于spring cache 重构,由国内资深人气工程师allen.chen,亲历打造轻量级redis缓存框架,旨在增加开发效率,和redis完美集合.
cacheChen的问世解决了spring 注解不具备的功能,解决缓存雪崩 穿透 问题,支持配置失效时间等强大功能难以想象。

用到了什么

  1. lombok https://projectlombok.org/
  2. spring boot
  3. jedis
<dependency>
    <groupId>redis.clients</groupId>
    <artifactId>jedis</artifactId>
    <version>2.9.0</version>
</dependency>

功能说明

  1. 缓存批量命中,批量处理
  2. 解决缓存穿透问题
  3. 解决缓存雪崩问题
  4. 支持配置缓存失效时间
  5. 实现缓存的读写分离,不同cacheName 配置不同的读写库
  6. 支持spring 4.0+ cache 注解功能

新增功能(20170813更新)

  • cacheName 同步key分布式锁功能
  1. 配置
 /**
  * 是否开启缓存key分布式锁
  */
   redisConfig.setDistributedLock(true);
  1. 参考
    http://git.oschina.net/iwantyou/cacheChen/blob/master/src/main/java/org/chen/cache/concurrent/distributed/DistributedLock.java

基本配置

@Configuration
public class Config {

    @Bean
    public BeanFactoryCacheOperationAdvisor cacheAdvisor(){
        BeanFactoryCacheOperationAdvisor advisor = new BeanFactoryCacheOperationAdvisor();
        advisor.setAdvice(cacheInterceptor());
        return advisor;
    }

    @Bean
    public CacheInterceptor cacheInterceptor(){
        return new CacheInterceptor();
    }

}
     /**
      * 创建cache manager
      * @return
      */
     @Bean
     public CacheManager createCacheManager(){
         RedisConfig redisConfig = new RedisConfig();
         redisConfig.setDefaultReadSourceName("read");
         redisConfig.setDefaultWriteSourceName("write");
 
         JedisPoolConfig config = new JedisPoolConfig();
 
         //连接耗尽时是否阻塞, false报异常,true 阻塞直到超时, 默认true
         config.setBlockWhenExhausted(true);
         //设置的逐出策略类名, 默认DefaultEvictionPolicy(当连接超过最大空闲时间,或连接数超过最大空闲连接数)
         config.setEvictionPolicyClassName("org.apache.commons.pool2.impl.DefaultEvictionPolicy");
         //是否启用pool的jmx管理功能, 默认true
         config.setJmxEnabled(true);
         //MBean ObjectName = new ObjectName("org.apache.commons.pool2:type=GenericObjectPool,name=" + "poolConfig" + i); 默认为"poolConfig", JMX不熟,具体不知道是干啥的...默认就好.
         config.setJmxNamePrefix("poolConfig");
         //是否启用后进先出, 默认true
         config.setLifo(true);
         //最大空闲连接数, 默认8个
         config.setMaxIdle(8);
         //最大连接数, 默认8个
         config.setMaxTotal(8);
         //获取连接时的最大等待毫秒数(如果设置为阻塞时BlockWhenExhausted),如果超时就抛异常, 小于零:阻塞不确定的时间,  默认-1
         config.setMaxWaitMillis(-1);
         //逐出连接的最小空闲时间 默认1800000毫秒(30分钟)
         config.setMinEvictableIdleTimeMillis(1800000);
         //最小空闲连接数, 默认0
         config.setMinIdle(0);
         //每次逐出检查时 逐出的最大数目 如果为负数就是 : 1/abs(n), 默认3
         config.setNumTestsPerEvictionRun(3);
         //对象空闲多久后逐出, 当空闲时间>该值 且 空闲连接>最大空闲数 时直接逐出,不再根据MinEvictableIdleTimeMillis判断  (默认逐出策略)
         config.setSoftMinEvictableIdleTimeMillis(1800000);
         //在获取连接的时候检查有效性, 默认false
         config.setTestOnBorrow(false);
         //在空闲时检查有效性, 默认false
         config.setTestWhileIdle(false);
         //逐出扫描的时间间隔(毫秒) 如果为负数,则不运行逐出线程, 默认-1
         config.setTimeBetweenEvictionRunsMillis(-1);
 //        poolConfig.setTestWhileIdle(true);
 //        poolConfig.setTestOnBorrow(true);
 
         RedisSource[] sources = new RedisSource[2];
         RedisSource readRedisSource = new RedisSource();
         readRedisSource.setSourceName("read");
         readRedisSource.setIp("192.168.0.118");
         readRedisSource.setPort(6379);
         readRedisSource.setPassword("chen");
         readRedisSource.setPoolConfig(config);
 
         RedisSource writeRedisSource = new RedisSource();
         writeRedisSource.setSourceName("write");
         writeRedisSource.setIp("192.168.0.118");
         writeRedisSource.setPort(6379);
         writeRedisSource.setPassword("chen");
         writeRedisSource.setPoolConfig(config);
 
         sources[0] = readRedisSource;
         sources[1] = writeRedisSource;
         redisConfig.setSources(sources);
 
         redisConfig.setUseLocalCache(true);
         /*
          * 分布式多服务 订阅本地缓存,配置此属性,必须开启本地缓存useLocalCache=true
          */
         redisConfig.setSubscribeLocalCacheChannel(true);
 
         /**
          * 是否开启缓存key分布式锁
          */
         redisConfig.setDistributedLock(true);
 
         CacheManager cacheManager = new CacheManager(redisConfig);
         return cacheManager;
     }

使用说明





使用案例

   //-------- Cacheable demo 开始 -------------------------------------------------------------
  
  
      @Cacheable(key = "#userId",cacheTime = 1000 * 50)
      public User get(Integer userId){
          return get(userId, null);
      }
  
      @Cacheable(key = "#userId",readSourceName = "test1",writeSourceName = "test2")
      public User getBySourceName(Integer userId){
          return get(userId, null);
      }
  
      @Cacheable(key = "#userId",sync = true)
      public User getSync(Integer userId){
          return get(userId, null);
      }
  
      @Caching(cacheable={@Cacheable(key = "#userId"),@Cacheable(key = "#email")})
      public User get(Integer userId, String email){
          User user = new User();
          user.setId(userId);
          user.setEmail(email);
          user.setName("测试");
          user.setCreateTime(new Date());
          return user;
      }
  
      @Cacheable(resultCacheKey = "id",batch = true)
      public ArrayList<User> batchGet(Collection<Integer> userIds){
          ArrayList<User> list = new ArrayList<>();
          for (Integer userId : userIds) {
              User user = new User();
              user.setId(userId);
              user.setName("批量测试");
              user.setCreateTime(new Date());
              list.add(user);
          }
  
          return list;
      }
  
      @Cacheable(resultCacheKey = "id",cacheNull = true,batch = true)
      public ArrayList<User> batchGetCacheNull(Collection<Integer> userIds){
  
          return batchGet(userIds);
      }
  
  
      @Cacheable(resultCacheKey = "id",slide = true,batch = true)
      public List<User> batchSlide(Collection<Integer> userIds){
          return  batchGet(userIds);
      }
  
  
      //-------- Cacheable demo 结束 -------------------------------------------------------------
  
  
      //--------   CachePut demo 开始 -------------------------------------------------------------
  
      @CachePut(cacheNames = "test",resultCacheKey = "id",batch = true)
      public List<User> batchPut(Collection<Integer> userIds){
          ArrayList<User> list = new ArrayList<>();
          for (Integer userId : userIds) {
              User user = new User();
              user.setId(userId);
              user.setName("批量put测试");
              user.setCreateTime(new Date());
              list.add(user);
          }
  
          return list;
      }
  
     @Caching(put={@CachePut(resultCacheKey = "userId"),@CachePut(resultCacheKey = "email")})
      public List<User> batchPutMuiltResultCacheKey(Collection<Integer> userIds){
          return batchPut(userIds);
      }
  
  
  
      //--------   CachePut demo 结束 -------------------------------------------------------------
  
      //--------   CacheEvict demo 开始 -------------------------------------------------------------
  
      @CacheEvict(key="#userId")
      public void remove(Integer userId){
  
      }
  
      @Caching(evict={@CacheEvict(key="#userId"),@CacheEvict(key="#email")})
      public void batchRemove(Integer userId,String email){
  
      }
  
      //--------   CacheEvict demo 结束 -------------------------------------------------------------

下载

http://git.oschina.net/iwantyou/cacheChen

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,702评论 18 139
  • /Library/Java/JavaVirtualMachines/jdk-9.jdk/Contents/Home...
    光剑书架上的书阅读 3,890评论 2 8
  • 相信我,这篇文章里只会出现借程序之名写的大众鸡汤。绝对与技术无关,任何人都能看的懂。如果非相关人士看后,能对程序代...
    阿丰在长春阅读 22,134评论 13 43
  • 在韩剧中,有一类女配角十分不招人待见。当然,女配角大多是不招人待见的,但这一类尤其被人唾弃。 她们通常任性,有想法...
    逆风的方向阅读 295评论 0 2
  • 2017一7月31篇一星期一一阴转小雨 下班回到家里,一顺问我今晚吃什么饭,我说感觉有点累,不想做让...
    一帆风顺平平安安阅读 199评论 0 0