我刚刚了解了Redis和Redisson。基本上我正在尝试使用Redis存储用于我的应用程序授权的AcessTokens / RefreshTokens。 所以我想存储具有到期时间的令牌。我使用 Spring Data Redis 来存储令牌,但没有Api使Map中的每个条目到期。我遇到了这篇文章Spring Data Redis Expire Key,因此抬头看了Redisson。我尝试了一个简单的maven java项目来测试过期。 这是pom.xml:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.bridgelabz</groupId>
<artifactId>redissonApp</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>jar</packaging>
<name>redissonApp</name>
<url>http://maven.apache.org</url>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>3.8.1</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.redisson</groupId>
<artifactId>redisson</artifactId>
<version>3.3.0</version>
</dependency>
</dependencies>
</project>
以下是令牌类
package com.bridgelabz.redissonApp;
public class Token {
private String accessToken;
private int id;
public Token() { }
public Token(String accessToken, int id) {
this.accessToken = accessToken;
this.id = id;
}
public String getAccessToken() {
return accessToken;
}
public void setAccessToken(String accessToken) {
this.accessToken = accessToken;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
@Override
public String toString() {
return "Token [accessToken=" + accessToken + ", id=" + id + "]";
}
}
这是我的演示App:
package com.bridgelabz.redissonApp;
import java.util.concurrent.TimeUnit;
import org.redisson.Redisson;
import org.redisson.api.LocalCachedMapOptions;
import org.redisson.api.RMapCache;
import org.redisson.api.RedissonClient;
import org.redisson.api.LocalCachedMapOptions.EvictionPolicy;
import org.redisson.config.Config;
public class App {
public static void main(String[] args) {
Config config = new Config();
config.useSingleServer().setAddress("127.0.0.1:6379");
// LocalCachedMapOptions localCachedMapOptions =
// LocalCachedMapOptions.defaults()
// .evictionPolicy(EvictionPolicy.LFU);
RedissonClient redisson = Redisson.create(config);
try {
RMapCache<Integer, Token> map = redisson.getMapCache("TestMap");
Token myToken = new Token("abc", 1);
map.put(1, myToken, 10, TimeUnit.SECONDS);
System.out.println("Stored value with key 1 is: " + map.get(1));
}
finally {
redisson.shutdown();
}
}
}
运行App.java后,我得到输出,得到输出为:
SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
Stored value with key 1 is: Token [accessToken=abc, id=1]
只需在10秒后评论put代码并运行应用程序,即可获得部分期望的结果:
LF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
Stored value with key 1 is: null
但是当我运行redis-cli时,我仍然在输出中得到值:
127.0.0.1:6379> hget TestMap 1
"\x00\x00\x00\x00\x00\x00\x00\x00H\x00\x00\x00\x00\x00\x00\x00{\"@class\":\"com.bridgelabz.redissonApp.Token\",\"accessToken\":\"abc\",\"id\":1}"
为什么密钥不会从redis中删除?仅供参考:一切都在我的本地机器上进行测试,仅包括redis。
答案 0 :(得分:4)
Redis不支持哈希开箱即用的单个元素驱逐。所以Redisson已经建立了自己的解决方案,并将其命名为MapCache
。
因此,使用MapCache
,您现在有两个级别的到期控制:Redis提供的关键级别和Redisson提供的字段级别。
在您的测试代码中:
RMapCache<Integer, Token> map = redisson.getMapCache("TestMap");
Token myToken = new Token("abc", 1);
map.put(1, myToken, 10, TimeUnit.SECONDS);
您已在哈希1
的字段TestMap
上设置了到期日期。这意味着哈希没有针对它设置的到期日期,而是针对其中的一个字段设置了到期日。因此,当您使用redis-cli
进行查找时,哈希仍然存在。当Redisson到期过程开始时,它最终会消失。
答案 1 :(得分:0)
我不知道推理,但是当我在Spring Web App中使用相同的代码但在简单的Java应用程序中无法工作时,密钥已过期。 这是TokenDaoImpl.java的代码:
import java.io.IOException;
import java.util.UUID;
import java.util.concurrent.TimeUnit;
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import org.apache.log4j.Logger;
import org.redisson.Redisson;
import org.redisson.api.RMapCache;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
import org.springframework.stereotype.Repository;
import com.bridgelabz.restApiDemo.entity.Token;
@Repository
public class TokenDaoImpl implements TokenDao {
Logger logger = Logger.getLogger(TokenDaoImpl.class);
RedissonClient redissonClient;
RMapCache<String, Token> tokenMap;
@PostConstruct
public void initRedisson() {
Config config = null;
try {
config = Config.fromJSON("{\n" + " \"singleServerConfig\":{\n"
+ " \"idleConnectionTimeout\":10000,\n" + " \"pingTimeout\":1000,\n"
+ " \"connectTimeout\":10000,\n" + " \"timeout\":3000,\n" + " \"retryAttempts\":3,\n"
+ " \"retryInterval\":1500,\n" + " \"reconnectionTimeout\":3000,\n"
+ " \"failedAttempts\":3,\n" + " \"password\":null,\n"
+ " \"subscriptionsPerConnection\":5,\n" + " \"clientName\":null,\n"
+ " \"address\": \"redis://127.0.0.1:6379\",\n"
+ " \"subscriptionConnectionMinimumIdleSize\":1,\n"
+ " \"subscriptionConnectionPoolSize\":50,\n" + " \"connectionMinimumIdleSize\":10,\n"
+ " \"connectionPoolSize\":64,\n" + " \"database\":0,\n"
+ " \"dnsMonitoring\":false,\n" + " \"dnsMonitoringInterval\":5000\n" + " },\n"
+ " \"threads\":0,\n" + " \"nettyThreads\":0,\n" + " \"codec\":null,\n"
+ " \"useLinuxNativeEpoll\":false\n" + "}");
} catch (IOException e1) {
logger.info("******Inside Config Catch");
e1.printStackTrace();
}
logger.info("**************Config Object" + config);
redissonClient = Redisson.create(config);
tokenMap = redissonClient.getMapCache("tokenMap");
}
@PreDestroy
public void redissonShutdown() {
redissonClient.shutdown();
}
public TokenDaoImpl() {
}
@Override
public Token generateToken(String tokenType, int uid) {
// first generate the token
String tokenValue = UUID.randomUUID().toString().replaceAll("-", "");
logger.info("******Generated access token is " + tokenValue);
Token token = new Token(tokenType, tokenValue, uid);
// save the token in redis cache with expiration depending upon tokenType
switch (tokenType) {
case "accessToken":
tokenMap.put(tokenValue, token, 15, TimeUnit.MINUTES);
break;
case "refreshToken":
tokenMap.put(tokenValue, token, 30, TimeUnit.MINUTES);
break;
case "forgotToken":
tokenMap.put(tokenValue, token, 30, TimeUnit.MINUTES);
break;
default:
logger.info("**********Please specify correct token type!");
break;
}
return token;
}
@Override
public boolean verifyToken(String tokenValue) {
Token token = tokenMap.get(tokenValue);
logger.info("******Token from redis for verification" + token);
if (token != null) {
return true;
}
return false;
}
}
以前,在运行Java App时,我有二个地图,我创建了一个地图,即TokenMap
,而新地图有MapName_session_timeout
,并且即使在指定的到期后,密钥也会保留。现在所有键都被正确删除。
答案 2 :(得分:0)
try {
RMapCache<Integer, Token> map = redisson.getMapCache("TestMap");
Token myToken = new Token("abc", 1);
map.put(1, myToken, 10, TimeUnit.SECONDS);
System.out.println("Stored value with key 1 is: " + map.get(1));
}
finally {
redisson.shutdown();
}
将项目放置到地图上后,您的应用已退出,然后当该项目过期时就没有机会为您的应用清理该项目。