spring cache 集成 cacheCloud redis

最近研究了一下cacheCloud,簡單的環境搭建了一下,覺得非常厲害,這裏就把一些集成客戶端的一些操作記錄下來,方便以後查看

  1. pom.xml文件配置
<!-- sohu的相關jar包 -->
 <dependency>
          <groupId>com.sohu.tv</groupId>
            <artifactId>cachecloud-jedis</artifactId>
            <version>${cachecloud-jedis}</version>
  </dependency>

<dependency>
            <groupId>com.sohu.tv</groupId>
            <artifactId>cachecloud-open-client-redis</artifactId>
            <version>${cachecloud-open-client-basic}</version>
            <exclusions>
                <exclusion>
                    <artifactId>jedis</artifactId>
                    <groupId>redis.clients</groupId>
                </exclusion>
            </exclusions>
        </dependency>

<!-- spring相關的jar包  我這裏的版本是 4.3.7.RELEASE -->
 <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${springframework.version}</version>
        </dependency>
 <!-- 工具 -->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.16</version>
        </dependency>
  1. 客戶端代碼
    接口類 , 這裏只定義了一些常用的操作,具體實現我就不貼了,這一塊可以根據你們自己的業務去做實現
import java.util.List;
import java.util.Map;

/**
 * 緩存接口定義
 *
 * @author Liukx
 * @create 2017-05-17 15:40
 * @email [email protected]
 **/
public interface ICacheClient {

    /**
     * 將一個數組key進行原子性相減
     *
     * @param key 鍵
     * @param num 減去的值
     * @return
     */
    Long decrby(String key, long num);

    /**
     * 將一個原子性的數值即你想那個累加
     * @param key   鍵
     * @param num   累加的值
     * @return
     */
    Long incrby(String key, long num);

    /**
     * 判斷key是否存在
     *
     * @param key
     * @return
     */
    boolean exsit(String key);

    /**
     * 設置一個鍵的有效時長
     *
     * @param key     鍵
     * @param seconds 有效時長 單位:秒
     */
    boolean expire(String key, int seconds);

    /**
     * 獲取一個key的有效時長
     *
     * @param key
     * @return
     */
    Long ttl(String key);

    /**
     * 添加一個普通值
     *
     * @param key   鍵
     * @param value 值
     */
    <T> void set(String key, T value);

    /**
     * 添加一個list的值
     *
     * @param key   鍵
     * @param list 值
     */
    <T> void setList(String key, List<T> list);

    /**
     * 添加一個map的值
     * @param key
     * @param map
     * @param <T>
     */
    <T> void setMap(String key, Map<String, T> map);

    /**
     * 添加一個值,併爲它設置一個有效時間
     *
     * @param key       鍵
     * @param value     值
     * @param validTime 有效時間  1秒=1000   -1 永久有效
     */
    void set(String key, Object value, int validTime);

    /**
     * 根據鍵獲取值
     *
     * @param key 鍵 - 標識
     * @return
     */
    <T>T get(String key, Class t);

    /**
     * 獲取list結果集
     * @param key
     * @param <T>
     * @return
     */
    <T> List<T> getList(String key, Class clazz);

    /**
     *  獲取map結果集
     * @param key
     * @param <T>
     * @return
     */
    <T> Map<String, T> getMap(String key, Class clazz);

    /**
     * 設置一個值到集合中,如果存在則返回key存在 則返回false
     *
     * @param key   鍵
     * @param value 值
     * @return
     */
    boolean setNX(String key, Object value);

    /**
     * 根據鍵刪除一個值
     *
     * @param key 鍵
     */
    void delete(String key);

    /**
     * 帶鎖的數據操作,例如當設置一個setnx發現鍵已經存在,則超時時間的範圍內阻塞嘗試,直到鎖被釋放,如果過了超時時間則表示失敗
     *
     * @param key     key
     * @param value   值
     * @param timeout 超時時間 單位秒
     * @return
     */
    boolean tryLock(String key, Object value, int timeout);

    /**
     * 刪除帶鎖的數據
     *
     * @param key 帶鎖的key
     */
    void unLock(String key);

    /**
     * <p>通過key 和offset 從指定的位置開始將原先value替換</p>
     * <p>下標從0開始,offset表示從offset下標開始替換</p>
     * <p>如果替換的字符串長度過小則會這樣</p>
     * <p>example:</p>
     * <p>value : [email protected]</p>
     * <p>str : abc </p>
     * <P>從下標7開始替換  則結果爲</p>
     * <p>RES : bigsea.abc.cn</p>
     * @param key
     * @param value
     * @param offset 下標位置
     * @return 返回替換後  value 的長度
     */
    Long setRange(String key, String value, int offset);

    /**
     * <p>通過下標 和key 獲取指定下標位置的 value</p>
     * @param key
     * @param startOffset 開始位置 從0 開始 負數表示從右邊開始截取
     * @param endOffset
     * @return 如果沒有返回null
     */
    String getRange(String key, int startOffset, int endOffset);

    /**
     * <p>通過批量的key獲取批量的value</p>
     * @param keys string數組 也可以是一個key
     * @return 成功返回value的集合, 失敗返回null的集合 ,異常返回空
     */
    List<String> mget(String... keys);

    /**
     * <p>批量的設置key:value,可以一個</p>
     * <p>example:</p>
     * <p>  obj.mset(new String[]{"key2","value1","key2","value2"})</p>
     * @param keysvalues
     * @return 成功返回OK 失敗 異常 返回 null
     *
     */
    String mset(String... keysvalues);


    /**
     * <p>批量的設置key:value,可以一個,如果key已經存在則會失敗,操作會回滾</p>
     * <p>example:</p>
     * <p>  obj.msetnx(new String[]{"key2","value1","key2","value2"})</p>
     * @param keysvalues
     * @return 成功返回1 失敗返回0
     */
    Long msetnx(String... keysvalues);

    /**
     * <p>通過key給Hash Field設置指定的值,如果key不存在,則先創建</p>
     * @param key
     * @param field 字段
     * @param value
     * @return 如果存在返回0 異常返回null
     */
    Long hset(String key, String field, String value);

    /**
     * <p>Sets field in the hash stored at key to value
     * 通過key給field設置指定的值,如果key不存在則先創建,如果field已經存在,返回0</p>
     * @param key
     * @param field
     * @param value
     * @return
     */
    public Long hsetnx(String key, String field, String value);

    /**
     * <p>通過key同時設置 hash的多個field</p>
     * @param key
     * @param hash
     * @return 返回OK 異常返回null
     */
    public String hmset(String key, Map<String, String> hash);

    /**
     * <p>通過key 和 field 獲取指定的 value</p>
     * @param key
     * @param field
     * @return 沒有返回null
     */
    public String hget(String key, String field);

    /**
     * <p>通過key 和 fields 獲取指定的value 如果沒有對應的value則返回null</p>
     * @param key
     * @param fields 可以是 一個String 也可以是 String數組
     * @return
     */
    public List<String> hmget(String key,String...fields);

    /**
     * <p>通過key向list頭部添加字符串</p>
     * @param key
     * @param strs 可以是一個string 也可以是string數組
     * @return 返回list的value個數
     */
    Long lpush(String key ,String...strs);

    /**
     * 獲取指定list
     * @param key 鍵
     * @param startIndex 開始下標
     * @param endIndex 結束下標  -1 代表獲取所有
     * @return
     */
    List<String> lrange(String key, int startIndex, int endIndex);

    /**
     * 獲取所有list
     * @param key 鍵
     * @return
     */
    List<String> lrange(String key);

    String get(String key);
}

然後就是實現Spring的cache接口的類,裏面摻雜了一些我們業務相關的代碼,你可以不用關注,大概瞭解意思就行了


import com.elab.cache.ICacheClient;
import com.elab.core.bean.Info;
import com.elab.core.utils.ObjectUtils;
import org.springframework.cache.Cache;
import org.springframework.cache.support.SimpleValueWrapper;

/**

 這個類是比較關鍵的,就是相當於你實現了spring的接口,只要將你的緩存交給spring,讓他去做一系列的事情就行了

*/
public class SystemCacheManage implements Cache {

    @Override
    public ValueWrapper putIfAbsent(Object key, Object value) {
        return null;
    }

    /**
     * Redis
     */
    private ICacheClient cacheClient;

    /**
     * 緩存名稱
     */
    private String name;

    /**
     * 超時時間
     */
    private int timeout;

    /*
     * (non-Javadoc)
     * @see org.springframework.cache.Cache#getName()
     */
    @Override
    public String getName() {
        return this.name;
    }

    /*
     * (non-Javadoc)
     * @see org.springframework.cache.Cache#getNativeCache()
     */
    @Override
    public Object getNativeCache() {
        return this.cacheClient;
    }

    /*
     * (non-Javadoc)
     * @see org.springframework.cache.Cache#get(java.lang.Object)
     */
    @Override
    public ValueWrapper get(Object key) {
        if (ObjectUtils.isEmpty(key)) {
            return null;
        } else {
            final String finalKey;
            if (key instanceof String) {
                finalKey = (String) key;
            } else {
                finalKey = key.toString();
            }
            Object object = cacheClient.get(finalKey, Info.class);
            return (object != null ? new SimpleValueWrapper(object) : null);
        }
    }

    /*
     * (non-Javadoc)
     * @see org.springframework.cache.Cache#get(java.lang.Object, java.lang.Class)
     */
    @SuppressWarnings("unchecked")
    @Override
    public <T> T get(Object key, Class<T> type) {
        if (ObjectUtils.isEmpty(key) || null == type) {
            return null;
        } else {
            final String finalKey;
            final Class<T> finalType = type;
            if (key instanceof String) {
                finalKey = (String) key;
            } else {
                finalKey = key.toString();
            }
            final Object object = cacheClient.get(finalKey, type);
            if (finalType != null && finalType.isInstance(object) && null != object) {
                return (T) object;
            } else {
                return null;
            }
        }
    }

    /*
     * (non-Javadoc)
     * @see org.springframework.cache.Cache#put(java.lang.Object, java.lang.Object)
     */
    @Override
    public void put(final Object key, final Object value) {
        if (ObjectUtils.isEmpty(key) || ObjectUtils.isEmpty(value)) {
            return;
        } else {
            final String finalKey;
            if (key instanceof String) {
                finalKey = (String) key;
            } else {
                finalKey = key.toString();
            }
            if (!ObjectUtils.isEmpty(finalKey)) {
                cacheClient.set(finalKey, value, timeout);
            }
        }
    }

    /*
     * 根據Key 刪除緩存
     */
    @Override
    public void evict(Object key) {
        if (null != key) {
            final String finalKey;
            if (key instanceof String) {
                finalKey = (String) key;
            } else {
                finalKey = key.toString();
            }
            if (!com.elab.core.utils.ObjectUtils.isEmpty(finalKey)) {
                cacheClient.delete(finalKey);
            }
        }
    }

    /*
     * 清除系統緩存
     */
    @Override
    public void clear() {
        // TODO Auto-generated method stub
        // redisTemplate.execute(new RedisCallback<String>() {
        // public String doInRedis(RedisConnection connection) throws DataAccessException {
        // connection.flushDb();
        // return "ok";
        // }
        // });
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getTimeout() {
        return timeout;
    }

    public void setTimeout(int timeout) {
        this.timeout = timeout;
    }

    public ICacheClient getCacheClient() {
        return cacheClient;
    }

    public void setCacheClient(ICacheClient cacheClient) {
        this.cacheClient = cacheClient;
    }
}

動態生成redis中的key生成策略,這個有需要就定義,沒有需要就不用,我這裏是根據類加方法名加參數構成的一個key

import com.alibaba.fastjson.JSON;
import org.springframework.cache.interceptor.KeyGenerator;

import java.lang.reflect.Method;

/**
 * 默認的緩存key生成策略
 *
 * @author Liukx
 * @create 2017-11-06 13:38
 * @email [email protected]
 **/
public class DefaultCacheKeyGenerator implements KeyGenerator {

    @Override
    public Object generate(Object target, Method method, Object... params) {
        String className = target.getClass().getSimpleName();
        String name = method.getName();
        String jsonParams = JSON.toJSONString(params);
        String cachekey = className + "_" + name + "_" + jsonParams;
        return cachekey;
    }
}

異常處理類,這裏異常只記錄了日誌沒有做什麼特殊的實現

/**
 * 緩存異常處理
 *
 * @author Liukx
 * @create 2017-11-06 17:58
 * @email [email protected]
 **/
public class ErrorCacheHandle implements CacheErrorHandler {

    Logger logger = LoggerFactory.getLogger(ErrorCacheHandle.class);

    public void handleCacheGetError(RuntimeException exception, Cache cache, Object key) {
        logger.error("‖‖‖‖‖‖‖‖‖緩存異常[handleCacheGetError]‖‖‖‖‖‖‖‖‖‖ cache : " + cache.getName() + " \t key : " + key.toString());
    }

    public void handleCachePutError(RuntimeException exception, Cache cache, Object key, Object value) {
        logger.error("‖‖‖‖‖‖‖‖‖緩存異常[handleCachePutError]‖‖‖‖‖‖‖‖‖‖ cache : " + cache.getName() + " \t key : " + key.toString() + "\t value : " + value);

    }

    public void handleCacheEvictError(RuntimeException exception, Cache cache, Object key) {
        logger.error("‖‖‖‖‖‖‖‖‖緩存異常[handleCacheEvictError]‖‖‖‖‖‖‖‖‖‖ cache : " + cache.getName() + " \t key : " + key.toString());
    }

    public void handleCacheClearError(RuntimeException exception, Cache cache) {
        logger.error("‖‖‖‖‖‖‖‖‖緩存異常[handleCacheClearError]‖‖‖‖‖‖‖‖‖‖ cache : " + cache.getName());
    }
}

接下來就是比較關鍵的配置文件定義了

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:cache="http://www.springframework.org/schema/cache"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans-4.1.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
     http://www.springframework.org/schema/cache
     http://www.springframework.org/schema/cache/spring-cache.xsd
    ">
    <!-- 默認生成的緩存key的實現策略 -->
    <bean id="defaultCacheKeyGenerator" class="com.elab.cache.spring.generator.DefaultCacheKeyGenerator"/>
    <!-- 異常處理類 -->
    <bean id="errorCacheHandle" class="com.elab.cache.spring.handle.ErrorCacheHandle" />

    <!-- 開啓緩存掃描註解包 -->
    <cache:annotation-driven  error-handler="errorCacheHandle" key-generator="defaultCacheKeyGenerator"/>
    
    <bean id="systemCacheManage" class="com.elab.cache.spring.manage.SystemCacheManage">
      <!-- 這裏是cacheCloud的一些代碼定義,其實就是它的實現類 -->
        <property name="cacheClient" ref="redisStandaloneClient"/>
        <property name="timeout" value="300"/>
        <property name="name" value="redisClient"/>
    </bean>

   <!-- 實現簡單的配置管理器 -->
    <bean id="simpleCacheManager" class="org.springframework.cache.support.SimpleCacheManager">
        <property name="caches" ref="systemCacheManage"/>
    </bean>

    <!-- 緩存管理器 -->
    <bean id="cacheManager"
          class="org.springframework.cache.support.CompositeCacheManager">
        <property name="cacheManagers">
            <list>
                <ref bean="simpleCacheManager"/>
            </list>
        </property>
        <!-- 當value找不到時,是否使用一個null[NoOpCacheManager]代替 -->
        <property name="fallbackToNoOpCache" value="true"/>
    </bean>
</beans>

如何測試?

在需要加入緩存的方法上面加入下面註解,需要注意的是

  • 第一個參數value 對應的就是配置文件中的systemCacheManage的name給定的值
  • 第二個參數就是你自己默認要指定的key,也就是redis中的key,可以是動態的,不填的話,就會調用上面默認的生成key的策略方法DefaultCacheKeyGenerator
  • 第三個參數就是當前緩存的方法必須滿足condition 中的條件才能被緩存!
// 會從緩存中查詢key,如果存在則不會進入該方法,不存在則進入,執行完之後進行緩存
@Cacheable(value = "redisClient", key = "'lkx_test_'+#redisModel.id", condition = "#redisModel.id != null")


//這個註解適用於DML操作的方法,表示每次都會執行該方法,並且緩存結果
    @CachePut(value = "redisClient", key = "'lkx_test_'+#redisModel.id")


    /**
    *  清除緩存的註解
     * allEntries : 是否清空所有緩存內容,缺省爲 false,如果指定爲 true,則方法調用後將立即清空所有緩存
     * beforeInvocation : 是否在方法執行前就清空,缺省爲 false,如果指定爲 true,則在方法還沒有執行的時候就清空緩存,缺省情況下,如果方法執行拋出異常,則不會清空緩存
     *
     * @param redisModel
     * @return
     */
    @CacheEvict(value = "redisClient", key = "'lkx_test_'+#redisModel.id", allEntries = false, beforeInvocation = false)

另外這些註解都是需要被掃描到的,所以類上面一定要有spring的掃描註解!!
比如@Service .... 寫的不是特別詳細,也是大概粗略的研究了下,不是特別深入,如果有不對的地方請指正,不懂的地方也可以提問....

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章