在做查詢資料的時候,遇到了需要設定資料在redis中第二天過期的問題,但是redis又沒有對應的API,就只好自己來解決了
計算出第二天凌晨與當前時間的時間差,將該時間差設定為redis的過期時間,就可以達到我們想要的效果
/** * 计算第二天凌晨与当前时间的时间差秒数 * @param * @return java.lang.Long * @author shy * @date 2021/3/12 18:10 */ public static Long getNowToNextDaySeconds() { Calendar cal = Calendar.getInstance(); cal.add(Calendar.DAY_OF_YEAR, 1); cal.set(Calendar.HOUR_OF_DAY, 0); cal.set(Calendar.SECOND, 0); cal.set(Calendar.MINUTE, 0); cal.set(Calendar.MILLISECOND, 0); return (cal.getTimeInMillis() - System.currentTimeMillis()) / 1000; }
拿到了時間差,剩下的基本上就沒什麼問題了。
附上Redis工具類別:
/** * 操作redis * @author shy * @date 2020/12/10 10:01 */ @Service public class RedisService { @Autowired private StringRedisTemplate stringRedisTemplate; @Autowired private RedisTemplate<String, Object> redisTemplate; /** * 判断String类型key是否存在 * * @param key * @return * @author shy * @date 2018年11月13日 下午1:40:37 */ public boolean hasStringKey(String key) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } return stringRedisTemplate.opsForValue().getOperations().hasKey(key); } /** * 判断String类型key是否存在 * * @param key * @return * @author shy * @date 2018年11月13日 下午1:43:51 */ public boolean nonStringKey(String key) { return !hasStringKey(key); } /** * 设置String类型key,String类型value,过期时间timeout,TimeUnit * * @param key * @param value * @param timeout * @param timeUnit * @author shy * @date 2018年12月10日13:53:38 */ public void setStringKey(String key, String value, Long timeout, TimeUnit timeUnit) { if (StringUtils.isBlank(key) || Objects.isNull(timeout)) { throw new EmptyParameterException(); } stringRedisTemplate.opsForValue().set(key, value, timeout, timeUnit); } public void setStringKey(String key, String value) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } stringRedisTemplate.opsForValue().set(key, value); } /** * 获取String类型value * * @param key * @return * @author shy * @date 2018年11月12日 下午7:09:31 */ public String getStringValue(String key) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } return stringRedisTemplate.opsForValue().get(key); } /** * 获取Key的过期时间 * * @param key * @return * @author shy * @date 2019年4月25日17:28:36 */ public Long getExpire(String key) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } return stringRedisTemplate.getExpire(key); } /** * 设置Key的过期时间 * * @param key * @return * @author shy * @date 2019年4月25日17:28:36 */ public Boolean setExpire(String key,Long timeout, TimeUnit timeUnit) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } return stringRedisTemplate.expire(key, timeout, timeUnit); } /** * value自增+n * @param key * @return * @author shy * @date 2019年4月8日15:54:30 */ public Long setIncrementValue(String key) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } return stringRedisTemplate.opsForValue().increment(key, 1L); } /** * 设置String类型key,Object类型value,过期时间timeout * * @param key * @param value * @param timeout * @author shy * @date 2018年12月10日13:54:07 */ public void setObjectKey(String key, Object value, Long timeout,TimeUnit time) { if (StringUtils.isBlank(key) || Objects.isNull(timeout)) { throw new EmptyParameterException(); } redisTemplate.opsForValue().set(key, value, timeout, time); } public void setObjectKey(String key, Object value) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } redisTemplate.opsForValue().set(key, value); } /** * 获取Object类型value * * @param key * @param clazz * @return * @author shy * @date 2019年11月6日10:01:30 */ @SuppressWarnings("unchecked") public <T> T getObjectValue(String key, Class<T> clazz) { if (StringUtils.isBlank(key)) { return null; } return (T) redisTemplate.opsForValue().get(key); } /** * 移除单个String类型key * * @param key * @author shy * @date 2018年11月13日 上午10:42:01 */ public void removeSingleStringKey(String key) { if (StringUtils.isBlank(key)) { throw new EmptyParameterException(); } stringRedisTemplate.opsForValue().getOperations().delete(key); } /** * 移除Collection<String>类型keys * * @param keys * @author shy * @date 2018年11月13日 下午3:15:16 */ public void removeMultiStringKey(Collection<String> keys) { if (CollectionUtils.isNotEmpty(keys)) { stringRedisTemplate.opsForValue().getOperations().delete(keys); } } /** * redis key 模糊查询 * @author shy * @date 2021年1月4日 上午11:21:45 * @param key * @return */ public Set<String> queryStringKeys(String key) { return redisTemplate.keys(key + "*"); } }
我們在使用redis時,一般會設定一個過期時間,當然也有不設定過期時間的,也就是永久不過期。
當我們設定了過期時間,redis是如何判斷是否過期,以及根據什麼策略來進行刪除的。
我們set key的時候,可以給一個expire time,就是過期時間,指定這個key比如說只能存活一個小時,假設你設定一批key存活一小時,那麼接下來一小時後,redis是如何對這批key進行刪除的?
答案是:定期刪除 惰性刪除。
所謂定期刪除是指redis預設每隔100ms就隨機抽取一些設定了過期時間的key,檢查其是否過期,如果過期就刪除。每隔100ms遍歷所有設定過期時間的key會導致效能大幅下降,請注意。 Redis實際上會在每100毫秒內隨機選擇一些key進行檢查和刪除。
但是問題是定期刪除可能會導致很多過期key到了時間並沒有被刪除,所以要惰性刪除,就是說在你獲取某個key的時候,redis會檢查一下,這個key如果設定了過期時間那麼是否過期了?如果過期了就會刪除。
透過上述兩種手段結合起來,保證過期的key一定會被幹掉。到了過期時間並不會自動刪除所有過期的key,因此過期後記憶體佔用並不會降低。
但其實這還是有問題的,如果定期刪除漏掉了很多過期key,然後沒有及時去查也就沒走惰性刪除,就會導致大量過期key堆積在內存裡耗費redis內存,這種情況如何處理?
答案是:走記憶體淘汰機制。
如果redis的記憶體佔用過多的時候,此時會進行一些淘汰,有以下一些策略:
#noeviction
:當記憶體不足以容納新寫入資料時,新寫入資料會報錯,這個實際場景一般不會使用。
allkeys-lru
:當記憶體不足以容納新寫入資料時,在鍵空間中,移除最少使用的key(這個是最常用的)
allkeys-random
:當記憶體不足以容納新寫入資料時,在鍵空間中,隨機移除某個key,這個一般用的比較少。
volatile-lru
:當記憶體不足以容納新寫入資料時,在設定了過期時間的鍵空間中,移除最近最少使用的key 。
volatile-random
:當記憶體不足以容納新寫入資料時,在設定了過期時間的鍵空間中,隨機移除某個key。
volatile-ttl
:當記憶體不足以容納新寫入資料時,在設定了過期時間的鍵空間中,有更早過期時間的key優先移除。
記憶體淘汰會觸發淘汰條件刪除某些key,這也是造成key沒有設定過期時間而遺失的原因。
以上是Java操作redis設定第二天凌晨過期的解決方案是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!