博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
cachecloud java_spring cache 集成 cacheCloud redis
阅读量:5149 次
发布时间:2019-06-13

本文共 10729 字,大约阅读时间需要 35 分钟。

最近研究了一下cacheCloud,简单的环境搭建了一下,觉得非常厉害,这里就把一些集成客户端的一些操作记录下来,方便以后查看

pom.xml文件配置

com.sohu.tv

cachecloud-jedis

${cachecloud-jedis}

com.sohu.tv

cachecloud-open-client-redis

${cachecloud-open-client-basic}

jedis

redis.clients

org.springframework

spring-core

${springframework.version}

com.alibaba

fastjson

1.2.16

客户端代码

接口类 , 这里只定义了一些常用的操作,具体实现我就不贴了,这一块可以根据你们自己的业务去做实现

import java.util.List;

import java.util.Map;

/**

* 缓存接口定义

*

* @author Liukx

* @create 2017-05-17 15:40

* @email liukx@elab-plus.com

**/

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 值

*/

void set(String key, T value);

/**

* 添加一个list的值

*

* @param key 键

* @param list 值

*/

void setList(String key, List list);

/**

* 添加一个map的值

* @param key

* @param map

* @param

*/

void setMap(String key, Map map);

/**

* 添加一个值,并为它设置一个有效时间

*

* @param key 键

* @param value 值

* @param validTime 有效时间 1秒=1000 -1 永久有效

*/

void set(String key, Object value, int validTime);

/**

* 根据键获取值

*

* @param key 键 - 标识

* @return

*/

T get(String key, Class t);

/**

* 获取list结果集

* @param key

* @param

* @return

*/

List getList(String key, Class clazz);

/**

* 获取map结果集

* @param key

* @param

* @return

*/

Map 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);

/**

*

通过key 和offset 从指定的位置开始将原先value替换

*

下标从0开始,offset表示从offset下标开始替换

*

如果替换的字符串长度过小则会这样

*

example:

*

value : bigsea@zto.cn

*

str : abc

*

从下标7开始替换 则结果为

*

RES : bigsea.abc.cn

* @param key

* @param value

* @param offset 下标位置

* @return 返回替换后 value 的长度

*/

Long setRange(String key, String value, int offset);

/**

*

通过下标 和key 获取指定下标位置的 value

* @param key

* @param startOffset 开始位置 从0 开始 负数表示从右边开始截取

* @param endOffset

* @return 如果没有返回null

*/

String getRange(String key, int startOffset, int endOffset);

/**

*

通过批量的key获取批量的value

* @param keys string数组 也可以是一个key

* @return 成功返回value的集合, 失败返回null的集合 ,异常返回空

*/

List mget(String... keys);

/**

*

批量的设置key:value,可以一个

*

example:

*

obj.mset(new String[]{"key2","value1","key2","value2"})

* @param keysvalues

* @return 成功返回OK 失败 异常 返回 null

*

*/

String mset(String... keysvalues);

/**

*

批量的设置key:value,可以一个,如果key已经存在则会失败,操作会回滚

*

example:

*

obj.msetnx(new String[]{"key2","value1","key2","value2"})

* @param keysvalues

* @return 成功返回1 失败返回0

*/

Long msetnx(String... keysvalues);

/**

*

通过key给Hash Field设置指定的值,如果key不存在,则先创建

* @param key

* @param field 字段

* @param value

* @return 如果存在返回0 异常返回null

*/

Long hset(String key, String field, String value);

/**

*

Sets field in the hash stored at key to value

* 通过key给field设置指定的值,如果key不存在则先创建,如果field已经存在,返回0

* @param key

* @param field

* @param value

* @return

*/

public Long hsetnx(String key, String field, String value);

/**

*

通过key同时设置 hash的多个field

* @param key

* @param hash

* @return 返回OK 异常返回null

*/

public String hmset(String key, Map hash);

/**

*

通过key 和 field 获取指定的 value

* @param key

* @param field

* @return 没有返回null

*/

public String hget(String key, String field);

/**

*

通过key 和 fields 获取指定的value 如果没有对应的value则返回null

* @param key

* @param fields 可以是 一个String 也可以是 String数组

* @return

*/

public List hmget(String key,String...fields);

/**

*

通过key向list头部添加字符串

* @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 lrange(String key, int startIndex, int endIndex);

/**

* 获取所有list

* @param key 键

* @return

*/

List 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 get(Object key, Class type) {

if (ObjectUtils.isEmpty(key) || null == type) {

return null;

} else {

final String finalKey;

final Class 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() {

// 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 liukx@elab-plus.com

**/

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 liukx@elab-plus.com

**/

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());

}

}

接下来就是比较关键的配置文件定义了

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

">

class="org.springframework.cache.support.CompositeCacheManager">

如何测试?

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

第一个参数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 .... 写的不是特别详细,也是大概粗略的研究了下,不是特别深入,如果有不对的地方请指正,不懂的地方也可以提问....

转载地址:http://podnv.baihongyu.com/

你可能感兴趣的文章
投标项目的脚本练习2
查看>>
第五次实验
查看>>
201521123107 《Java程序设计》第9周学习总结
查看>>
runtime的基本应用
查看>>
localStorage,最简单的历史记录
查看>>
关于scrollTop的那些事
查看>>
Caroline--chochukmo
查看>>
算法导论笔记 第8章 线性时间排序
查看>>
利用jquery的contains实现搜索功能
查看>>
Xcode 6.2 beta 3 太难下载!下载了,不敢独享
查看>>
并发编程
查看>>
Django框架(七)
查看>>
Linux文件系统概述
查看>>
ffmpeg-php
查看>>
别把淘宝客当傻子
查看>>
MySQL 数据库性能优化之SQL优化
查看>>
Bootstrap
查看>>
C语言错误: HEAP CORRUPTION DETECTED
查看>>
bzoj1102 [POI2007]山峰和山谷Grz
查看>>
【Java基础】Java类的加载和对象创建流程的详细分析
查看>>