package com.ruoyi.common.utils;
import com.google.common.collect.Lists;
import com.ruoyi.common.utils.spring.SpringUtils;
import lombok.AccessLevel;
import lombok.NoArgsConstructor;
import org.redisson.api.*;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;
/**
* redis 工具类
*
* @author Lion Li
* @version 3.1.0 新增
*/
@NoArgsConstructor(access = AccessLevel.PRIVATE)
@SuppressWarnings(value = {"unchecked", "rawtypes"})
public class RedisUtils {
private static RedissonClient client = SpringUtils.getBean(RedissonClient.class);
/**
* 陿µ
*
* @param key 陿µkey
* @param rateType 陿µç±»åž‹
* @param rate 速率
* @param rateInterval 速率间隔
* @return -1 表示失败
*/
public static long rateLimiter(String key, RateType rateType, int rate, int rateInterval) {
RRateLimiter rateLimiter = client.getRateLimiter(key);
rateLimiter.trySetRate(rateType, rate, rateInterval, RateIntervalUnit.SECONDS);
if (rateLimiter.tryAcquire()) {
return rateLimiter.availablePermits();
} else {
return -1L;
}
}
/**
* 获å–实例id
*/
public static String getClientId() {
return client.getId();
}
/**
* å‘å¸ƒé€šé“æ¶ˆæ¯
*
* @param channelKey 通é“key
* @param msg å‘逿•°æ®
* @param consumer 自定义处ç†
*/
public static <T> void publish(String channelKey, T msg, Consumer<T> consumer) {
RTopic topic = client.getTopic(channelKey);
topic.publish(msg);
consumer.accept(msg);
}
public static <T> void publish(String channelKey, T msg) {
RTopic topic = client.getTopic(channelKey);
topic.publish(msg);
}
/**
* è®¢é˜…é€šé“æŽ¥æ”¶æ¶ˆæ¯
*
* @param channelKey 通é“key
* @param clazz 消æ¯ç±»åž‹
* @param consumer 自定义处ç†
*/
public static <T> void subscribe(String channelKey, Class<T> clazz, Consumer<T> consumer) {
RTopic topic = client.getTopic(channelKey);
topic.addListener(clazz, (channel, msg) -> consumer.accept(msg));
}
/**
* 缓å˜åŸºæœ¬çš„对象,Integerã€Stringã€å®žä½“ç±»ç‰
*
* @param key 缓å˜çš„键值
* @param value 缓å˜çš„值
*/
public static <T> void setCacheObject(final String key, final T value) {
setCacheObject(key, value, false);
}
/**
* 缓å˜åŸºæœ¬çš„对象,ä¿ç•™å½“å‰å¯¹è±¡ TTL 有效期
*
* @param key 缓å˜çš„键值
* @param value 缓å˜çš„值
* @param isSaveTtl 是å¦ä¿ç•™TTL有效期(例如: set之å‰ttl剩余90 set之åŽè¿˜æ˜¯ä¸º90)
* @since Redis 6.X 以上使用 setAndKeepTTL 兼容 5.X 方案
*/
public static <T> void setCacheObject(final String key, final T value, final boolean isSaveTtl) {
RBucket<Object> bucket = client.getBucket(key);
if (isSaveTtl) {
try {
bucket.setAndKeepTTL(value);
} catch (Exception e) {
long timeToLive = bucket.remainTimeToLive();
bucket.set(value);
bucket.expire(timeToLive, TimeUnit.MILLISECONDS);
}
} else {
bucket.set(value);
}
}
/**
* 缓å˜åŸºæœ¬çš„对象,Integerã€Stringã€å®žä½“ç±»ç‰
*
* @param key 缓å˜çš„键值
* @param value 缓å˜çš„值
* @param timeout æ—¶é—´
* @param timeUnit 时间颗粒度
*/
public static <T> void setCacheObject(final String key, final T value, final long timeout, final TimeUnit timeUnit) {
RBucket<T> result = client.getBucket(key);
result.set(value);
result.expire(timeout, timeUnit);
}
/**
* 设置有效时间
*
* @param key Redisé”®
* @param timeout è¶…æ—¶æ—¶é—´
* @return true=设置æˆåŠŸï¼›false=设置失败
*/
public static boolean expire(final String key, final long timeout) {
return expire(key, timeout, TimeUnit.SECONDS);
}
/**
* 设置有效时间
*
* @param key Redisé”®
* @param timeout è¶…æ—¶æ—¶é—´
* @param unit æ—¶é—´å•ä½
* @return true=设置æˆåŠŸï¼›false=设置失败
*/
public static boolean expire(final String key, final long timeout, final TimeUnit unit) {
RBucket rBucket = client.getBucket(key);
return rBucket.expire(timeout, unit);
}
/**
* 获得缓å˜çš„基本对象。
*
* @param key 缓å˜é”®å€¼
* @return 缓å˜é”®å€¼å¯¹åº”的数æ®
*/
public static <T> T getCacheObject(final String key) {
RBucket<T> rBucket = client.getBucket(key);
return rBucket.get();
}
/**
* 获得keyå‰©ä½™å˜æ´»æ—¶é—´
*
* @param key 缓å˜é”®å€¼
* @return å‰©ä½™å˜æ´»æ—¶é—´
*/
public static <T> long getTimeToLive(final String key) {
RBucket<T> rBucket = client.getBucket(key);
return rBucket.remainTimeToLive();
}
/**
* åˆ é™¤å•个对象
*
* @param key
*/
public static boolean deleteObject(final String key) {
return client.getBucket(key).delete();
}
/* */
/**
* åˆ é™¤é›†åˆå¯¹è±¡
*
* @param collection 多个对象
* @return
*/
public static void deleteObject(final Collection collection) {
RBatch batch = client.createBatch();
collection.forEach(t -> {
batch.getBucket(t.toString()).deleteAsync();
});
batch.execute();
}
/**
* 缓å˜Listæ•°æ®
*
* @param key 缓å˜çš„键值
* @param dataList 待缓å˜çš„Listæ•°æ®
* @return 缓å˜çš„对象
*/
public static <T> boolean setCacheList(final String key, final List<T> dataList) {
RList<T> rList = client.getList(key);
return rList.addAll(dataList);
}
/**
* 获得缓å˜çš„list对象
*
* @param key 缓å˜çš„键值
* @return 缓å˜é”®å€¼å¯¹åº”的数æ®
*/
public static <T> List<T> getCacheList(final String key) {
RList<T> rList = client.getList(key);
return rList.readAll();
}
/**
* 缓å˜Set
*
* @param key 缓å˜é”®å€¼
* @param dataSet 缓å˜çš„æ•°æ®
* @return ç¼“å˜æ•°æ®çš„对象
*/
public static <T> boolean setCacheSet(final String key, final Set<T> dataSet) {
RSet<T> rSet = client.getSet(key);
return rSet.addAll(dataSet);
}
/**
* 获得缓å˜çš„set
*
* @param key
* @return
*/
public static <T> Set<T> getCacheSet(final String key) {
RSet<T> rSet = client.getSet(key);
return rSet.readAll();
}
/**
* 缓å˜Map
*
* @param key
* @param dataMap
*/
public static <T> void setCacheMap(final String key, final Map<String, T> dataMap) {
if (dataMap != null) {
RMap<String, T> rMap = client.getMap(key);
rMap.putAll(dataMap);
}
}
/**
* 获得缓å˜çš„Map
*
* @param key
* @return
*/
public static <T> Map<String, T> getCacheMap(final String key) {
RMap<String, T> rMap = client.getMap(key);
return rMap.getAll(rMap.keySet());
}
/**
* å¾€Hashä¸å˜å…¥æ•°æ®
*
* @param key Redisé”®
* @param hKey Hashé”®
* @param value 值
*/
public static <T> void setCacheMapValue(final String key, final String hKey, final T value) {
RMap<String, T> rMap = client.getMap(key);
rMap.put(hKey, value);
}
/**
* 获å–Hashä¸çš„æ•°æ®
*
* @param key Redisé”®
* @param hKey Hashé”®
* @return Hashä¸çš„对象
*/
public static <T> T getCacheMapValue(final String key, final String hKey) {
RMap<String, T> rMap = client.getMap(key);
return rMap.get(hKey);
}
/**
* åˆ é™¤Hashä¸çš„æ•°æ®
*
* @param key Redisé”®
* @param hKey Hashé”®
* @return Hashä¸çš„对象
*/
public static <T> T delCacheMapValue(final String key, final String hKey) {
RMap<String, T> rMap = client.getMap(key);
return rMap.remove(hKey);
}
/**
* 获å–多个Hashä¸çš„æ•°æ®
*
* @param key Redisé”®
* @param hKeys Hash键集åˆ
* @return Hash对象集åˆ
*/
public static <K, V> Map<K, V> getMultiCacheMapValue(final String key, final Set<K> hKeys) {
RMap<K, V> rMap = client.getMap(key);
return rMap.getAll(hKeys);
}
/**
* 获得缓å˜çš„基本对象列表
*
* @param pattern å—符串å‰ç¼€
* @return 对象列表
*/
public static Collection<String> keys(final String pattern) {
Iterable<String> iterable = client.getKeys().getKeysByPattern(pattern);
return Lists.newArrayList(iterable);
}
}