springboot配置基於redis的緩存(二)
上文(一)描寫了基於註解的緩存,其實底層邏輯就是redis的get 、set操作,將操作結果返回redis中。因此我們可以使用RedisUtils來進行自己設置,比如序列化的設置,key的設置都更加靈活一些。
在RedisConfig中添加redisTemplate的bean和 RedisUtils的初始化構造
@Bean
@ConditionalOnMissingBean(name = "redisTemplate")
public RedisTemplate<String, Object> redisTemplate(
RedisConnectionFactory redisConnectionFactory) {
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(Object.class);
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
// 解決jackson2無法反序列化LocalDateTime的問題
om.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
om.registerModule(new JavaTimeModule());
om.activateDefaultTyping(om.getPolymorphicTypeValidator(), ObjectMapper.DefaultTyping.NON_FINAL);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
RedisTemplate<String, Object> template = new RedisTemplate<String, Object>();
template.setConnectionFactory(redisConnectionFactory);
// template.setKeySerializer(jackson2JsonRedisSerializer);
template.setKeySerializer(new StringRedisSerializer());
template.setValueSerializer(jackson2JsonRedisSerializer);
template.setHashKeySerializer(jackson2JsonRedisSerializer);
template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.afterPropertiesSet();
return template;
}
@Bean
@ConditionalOnMissingBean(StringRedisTemplate.class)
public StringRedisTemplate stringRedisTemplate(
RedisConnectionFactory redisConnectionFactory) {
StringRedisTemplate template = new StringRedisTemplate();
template.setConnectionFactory(redisConnectionFactory);
return template;
}
@Bean
IGlobalCache cache(RedisTemplate redisTemplate) {
return new RedisUtils(redisTemplate);
}
-
接口IGlobalCache
public interface IGlobalCache { boolean flushdb(); boolean deleteStratKey(String startKey); /** * 指定緩存失效時間 * * @param key 鍵 * @param time 時間(秒) * @return */ boolean expire(String key, long time); /** * @param key 鍵 不能爲null * @return 時間(秒) 返回0代表爲永久有效 */ long getExpire(String key); /** * 判斷key是否存在 * * @param key 鍵 * @return true 存在 false不存在 */ boolean hasKey(String key); /** * 刪除緩存 * * @param key 可以傳一個值 或多個 */ void del(String... key); // ============================String============================= /** * 普通緩存獲取 * * @param key 鍵 * @return 值 */ Object get(String key); /** * 普通緩存放入 * * @param key 鍵 * @param value 值 * @return true成功 false失敗 */ boolean set(String key, Object value); /** * 普通緩存放入並設置時間 * * @param key 鍵 * @param value 值 * @param time 時間(秒) time要大於0 如果time小於等於0 將設置無限期 * @return true成功 false 失敗 */ boolean set(String key, Object value, long time); /** * 遞增 * * @param key 鍵 * @param delta 要增加幾(大於0) * @return */ long incr(String key, long delta); /** * 遞減 * * @param key 鍵 * @param delta 要減少幾(小於0) * @return */ long decr(String key, long delta); /** * HashGet * * @param key 鍵 不能爲null * @param item 項 不能爲null * @return 值 */ Object hget(String key, String item); /** * 獲取hashKey對應的所有鍵值 * * @param key 鍵 * @return 對應的多個鍵值 */ Map<Object, Object> hmget(String key); /** * HashSet * * @param key 鍵 * @param map 對應多個鍵值 * @return true 成功 false 失敗 */ boolean hmset(String key, Map<String, Object> map); /** * HashSet 並設置時間 * * @param key 鍵 * @param map 對應多個鍵值 * @param time 時間(秒) * @return true成功 false失敗 */ boolean hmset(String key, Map<String, Object> map, long time); /** * 向一張hash表中放入數據,如果不存在將創建 * * @param key 鍵 * @param item 項 * @param value 值 * @return true 成功 false失敗 */ boolean hset(String key, String item, Object value); /** * 向一張hash表中放入數據,如果不存在將創建 * * @param key 鍵 * @param item 項 * @param value 值 * @param time 時間(秒) 注意:如果已存在的hash表有時間,這裏將會替換原有的時間 * @return true 成功 false失敗 */ boolean hset(String key, String item, Object value, long time); /** * 刪除hash表中的值 * * @param key 鍵 不能爲null * @param item 項 可以使多個 不能爲null */ void hdel(String key, Object... item); /** * 判斷hash表中是否有該項的值 * * @param key 鍵 不能爲null * @param item 項 不能爲null * @return true 存在 false不存在 */ boolean hHasKey(String key, String item); /** * hash遞增 如果不存在,就會創建一個 並把新增後的值返回 * * @param key 鍵 * @param item 項 * @param by 要增加幾(大於0) * @return */ double hincr(String key, String item, double by); /** * hash遞減 * * @param key 鍵 * @param item 項 * @param by 要減少記(小於0) * @return */ double hdecr(String key, String item, double by); /** * 根據key獲取Set中的所有值 * * @param key 鍵 * @return */ Set<Object> sGet(String key); /** * 根據value從一個set中查詢,是否存在 * * @param key 鍵 * @param value 值 * @return true 存在 false不存在 */ boolean sHasKey(String key, Object value); /** * 將數據放入set緩存 * * @param key 鍵 * @param values 值 可以是多個 * @return 成功個數 */ long sSet(String key, Object... values); /** * 將set數據放入緩存 * * @param key 鍵 * @param time 時間(秒) * @param values 值 可以是多個 * @return 成功個數 */ long sSetAndTime(String key, long time, Object... values); /** * 獲取set緩存的長度 * * @param key 鍵 * @return */ long sGetSetSize(String key); /** * 移除值爲value的 * * @param key 鍵 * @param values 值 可以是多個 * @return 移除的個數 */ long setRemove(String key, Object... values); /** * 獲取list緩存的內容 * * @param key 鍵 * @param start 開始 * @param end 結束 0 到 -1代表所有值 * @return */ List<Object> lGet(String key, long start, long end); /** * 獲取list緩存的長度 * * @param key 鍵 * @return */ long lGetListSize(String key); /** * 通過索引 獲取list中的值 * * @param key 鍵 * @param index 索引 index>=0時, 0 表頭,1 第二個元素,依次類推;index<0時,-1,表尾,-2倒數第二個元素,依次類推 * @return */ Object lGetIndex(String key, long index); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @return */ boolean lSet(String key, Object value); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @return */ boolean lSet(String key, Object value, long time); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @param time 時間(秒) * @return */ boolean lSetAll(String key, List<Object> value); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @param time 時間(秒) * @return */ boolean lSetAll(String key, List<Object> value, long time); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @return */ boolean rSet(String key, Object value); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @param time 時間(秒) * @return */ boolean rSet(String key, Object value, long time); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @return */ boolean rSetAll(String key, List<Object> value); /** * 將list放入緩存 * * @param key 鍵 * @param value 值 * @param time 時間(秒) * @return */ boolean rSetAll(String key, List<Object> value, long time); /** * 根據索引修改list中的某條數據 * * @param key 鍵 * @param index 索引 * @param value 值 * @return */ boolean lUpdateIndex(String key, long index, Object value); /** * 移除N個值爲value * * @param key 鍵 * @param count 移除多少個 * @param value 值 * @return 移除的個數 */ long lRemove(String key, long count, Object value); /** * 從redis集合中移除[start,end]之間的元素 * * @param key * @param stard * @param end * @return */ void rangeRemove(String key, Long stard, Long end); /** * 返回當前redisTemplate * * @return */ RedisTemplate getRedisTemplate(); }
-
實現接口
@Getter
@AllArgsConstructor
public final class RedisUtils implements IGlobalCache {
private RedisTemplate<String, Object> redisTemplate;
@Override
public boolean flushdb() {
try {
redisTemplate.execute((RedisCallback) connection -> {
connection.flushDb();
return null;
});
// Set<String> keys = redisTemplate.keys("*");
// redisTemplate.delete(keys);
return true;
}
catch (Exception ex){
throw ex;
}
}
@Override
public boolean deleteStratKey(String startKey) {
try {
Set<String> keys = redisTemplate.keys(startKey);
redisTemplate.delete(keys);
return true;
}
catch (Exception ex){
throw ex;
}
}
@Override
public boolean expire(String key, long time) {
try {
if (time > 0) {
redisTemplate.expire(key, time, TimeUnit.SECONDS);
}
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public long getExpire(String key) {
return redisTemplate.getExpire(key, TimeUnit.SECONDS);
}
@Override
public boolean hasKey(String key) {
try {
return redisTemplate.hasKey(key);
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
@SuppressWarnings("unchecked")
public void del(String... key) {
if (key != null && key.length > 0) {
if (key.length == 1) {
redisTemplate.delete(key[0]);
} else {
redisTemplate.delete((Collection<String>) CollectionUtils.arrayToList(key));
}
}
}
@Override
public Object get(String key) {
return key == null ? null : redisTemplate.opsForValue().get(key);
}
@Override
public boolean set(String key, Object value) {
try {
redisTemplate.opsForValue().set(key, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean set(String key, Object value, long time) {
try {
if (time > 0) {
redisTemplate.opsForValue().set(key, value, time, TimeUnit.SECONDS);
} else {
set(key, value);
}
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public long incr(String key, long delta) {
if (delta < 0) {
throw new RuntimeException("遞增因子必須大於0");
}
return redisTemplate.opsForValue().increment(key, delta);
}
@Override
public long decr(String key, long delta) {
if (delta < 0) {
throw new RuntimeException("遞減因子必須大於0");
}
return redisTemplate.opsForValue().increment(key, -delta);
}
@Override
public Object hget(String key, String item) {
return redisTemplate.opsForHash().get(key, item);
}
@Override
public Map<Object, Object> hmget(String key) {
return redisTemplate.opsForHash().entries(key);
}
@Override
public boolean hmset(String key, Map<String, Object> map) {
try {
redisTemplate.opsForHash().putAll(key, map);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean hmset(String key, Map<String, Object> map, long time) {
try {
redisTemplate.opsForHash().putAll(key, map);
if (time > 0) {
expire(key, time);
}
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean hset(String key, String item, Object value) {
try {
redisTemplate.opsForHash().put(key, item, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean hset(String key, String item, Object value, long time) {
try {
redisTemplate.opsForHash().put(key, item, value);
if (time > 0) {
expire(key, time);
}
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public void hdel(String key, Object... item) {
redisTemplate.opsForHash().delete(key, item);
}
@Override
public boolean hHasKey(String key, String item) {
return redisTemplate.opsForHash().hasKey(key, item);
}
@Override
public double hincr(String key, String item, double by) {
return redisTemplate.opsForHash().increment(key, item, by);
}
@Override
public double hdecr(String key, String item, double by) {
return redisTemplate.opsForHash().increment(key, item, -by);
}
@Override
public Set<Object> sGet(String key) {
try {
return redisTemplate.opsForSet().members(key);
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
@Override
public boolean sHasKey(String key, Object value) {
try {
return redisTemplate.opsForSet().isMember(key, value);
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public long sSet(String key, Object... values) {
try {
return redisTemplate.opsForSet().add(key, values);
} catch (Exception e) {
e.printStackTrace();
return 0;
}
}
@Override
public long sSetAndTime(String key, long time, Object... values) {
try {
Long count = redisTemplate.opsForSet().add(key, values);
if (time > 0) {
expire(key, time);
}
return count;
} catch (Exception e) {
e.printStackTrace();
return 0;
}
}
@Override
public long sGetSetSize(String key) {
try {
return redisTemplate.opsForSet().size(key);
} catch (Exception e) {
e.printStackTrace();
return 0;
}
}
@Override
public long setRemove(String key, Object... values) {
try {
Long count = redisTemplate.opsForSet().remove(key, values);
return count;
} catch (Exception e) {
e.printStackTrace();
return 0;
}
}
@Override
public List<Object> lGet(String key, long start, long end) {
try {
return redisTemplate.opsForList().range(key, start, end);
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
@Override
public long lGetListSize(String key) {
try {
return redisTemplate.opsForList().size(key);
} catch (Exception e) {
e.printStackTrace();
return 0;
}
}
@Override
public Object lGetIndex(String key, long index) {
try {
return redisTemplate.opsForList().index(key, index);
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
@Override
public boolean lSetAll(String key, List<Object> value) {
try {
redisTemplate.opsForList().leftPushAll(key, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean lSet(String key, Object value) {
try {
redisTemplate.opsForList().leftPush(key, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean lSet(String key, Object value, long time) {
try {
redisTemplate.opsForList().leftPush(key, value);
if (time > 0) {
expire(key, time);
}
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean lSetAll(String key, List<Object> value, long time) {
try {
redisTemplate.opsForList().leftPushAll(key, value);
if (time > 0)
expire(key, time);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean rSet(String key, Object value) {
try {
redisTemplate.opsForList().rightPush(key, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean rSet(String key, Object value, long time) {
try {
redisTemplate.opsForList().rightPush(key, value);
if (time > 0) {
expire(key, time);
}
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean rSetAll(String key, List<Object> value) {
try {
redisTemplate.opsForList().rightPushAll(key, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean rSetAll(String key, List<Object> value, long time) {
try {
redisTemplate.opsForList().rightPushAll(key, value);
if (time > 0)
expire(key, time);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public boolean lUpdateIndex(String key, long index, Object value) {
try {
redisTemplate.opsForList().set(key, index, value);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
@Override
public long lRemove(String key, long count, Object value) {
try {
Long remove = redisTemplate.opsForList().remove(key, count, value);
return remove;
} catch (Exception e) {
e.printStackTrace();
return 0;
}
}
@Override
public void rangeRemove(String key, Long stard, Long end) {
try {
redisTemplate.opsForList().trim(key, stard, end);
} catch (Exception e) {
e.printStackTrace();
}
}
}
- 使用緩存
private static final String ONE_DAY_CONTENT = "SYNCHRONY_CONTENT";
@Autowired
ContentServer contentServer;
// 獲取
@GetMapping(value = "getCacheContent")
public Object getCacheContent(HttpServletRequest request) {
HttpSession session = request.getSession(false);
if (session != null && session.getAttribute("user") != null) {
UserPo user = (UserPo) session.getAttribute("user");
List<Object> list = globalCache.lGet(ONE_DAY_CONTENT + user.getUserName(), 0, -1);
return list;
} else {
return new ArrayList<>();
}
}
// 添加
globalCache.lSet(ONE_DAY_CONTENT + user.getUserName(), content);
其實就是簡單的redis操作,key的設置和過期時間都可以在交互時直接設置。本質還是操作redis的基本類型