具有复杂键的 Spring Data Redis 存储库
SpringData Redis Repository with complex key
我们尝试在我们的项目中使用 Spring 数据 CrudRepository
来为我们的域对象提供持久性。
一开始,我选择了 REDIS 作为后端,因为在第一次使用 CrudRepository<ExperimentDomainObject, String>
的实验中,得到它 运行 很容易。
当试图将它放入我们的生产代码中时,事情变得更加复杂,因为这里我们的域对象没有必要使用简单类型作为键,所以存储库是 CrudRepository<TestObject, ObjectId>
。
现在我得到了异常:
No converter found capable of converting from type [...ObjectId] to type [byte[]]
搜索此异常, 这导致我对 RedisTemplate
配置进行了未受过教育的试验。 (对于我的实验,我使用的是 emdedded-redis)
我的想法是,提供 RedisTemplate<Object, Object>
而不是 RedisTemplate<String, Object>
以允许使用 Jackson2JsonRedisSerializer
来完成 keySerializer 的工作。
仍然调用 testRepository.save(testObject)
失败。
请看我的代码:
我有 public 个字段,为了本示例的简洁,省略了导入。如果需要它们(使其成为 MVCE),我会很乐意提供它们。请给我留言。
dependencies:
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-data-redis'
implementation 'org.springframework.boot:spring-boot-starter-web'
compileOnly 'org.projectlombok:lombok'
annotationProcessor 'org.projectlombok:lombok'
implementation group: 'redis.clients', name: "jedis", version: '2.9.0'
implementation group: 'it.ozimov', name: 'embedded-redis', version: '0.7.2'
}
RedisConfiguration:
@Configuration
@EnableRedisRepositories
public class RedisConfiguration {
@Bean
JedisConnectionFactory jedisConnectionFactory() {
return new JedisConnectionFactory();
}
@Bean
public RedisTemplate<Object, Object> redisTemplate() {
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
final RedisTemplate<Object, Object> template = new RedisTemplate<>();
template.setConnectionFactory(jedisConnectionFactory());
template.setDefaultSerializer(jackson2JsonRedisSerializer);
template.setKeySerializer(jackson2JsonRedisSerializer);
template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.setValueSerializer(jackson2JsonRedisSerializer);
template.setEnableDefaultSerializer(true);
return template;
}
}
TestObject
@RedisHash("test")
public class TestObject
{
@Id public ObjectId testId;
public String value;
public TestObject(ObjectId id, String value)
{
this.testId = id;
this.value = value; // In experiment this is: "magic"
}
}
ObjectId
@EqualsAndHashCode
public class ObjectId {
public String creator; // In experiment, this is "me"
public String name; // In experiment, this is "fool"
}
TestRepository
@Repository
public interface TestRepository extends CrudRepository<TestObject, ObjectId>
{
}
EmbeddedRedisConfiguration
@Configuration
public class EmbeddedRedisConfiguration
{
private final redis.embedded.RedisServer redisServer;
EmbeddedRedisConfiguration(RedisProperties redisProperties)
{
this.redisServer = new redis.embedded.RedisServer(redisProperties.getPort());
}
@PostConstruct
public void init()
{
redisServer.start();
}
@PreDestroy
public void shutdown()
{
redisServer.stop();
}
}
Application:
@SpringBootApplication
public class ExperimentApplication
{
public static void main(String[] args)
{
SpringApplication.run(ExperimentApplication.class, args);
}
}
不是想要的答案:
当然,我可能会引入一些特殊的ID,它是一种简单的数据类型,例如我使用 jacksons ObjectMapper 手动构建的 JSON-String,然后使用 CrudRepository<TestObject, String>
.
同时我也试过:
RedisTemplate<String, String>
RedisTemplate<String, Object>
- 自动装配
RedisTemplate
并设置其默认序列化器
- 正在注册
Converter<ObjectId, byte[]>
- 自动装配
ConverterRegistry
- 自动装配
GenericConversionService
但显然他们错了。
基本上,Redis 存储库在底层使用 RedisKeyValueTemplate
将数据存储为键 (Id) 和值对。所以你配置的 RedisTemplate
除非你直接使用它,否则不会起作用。
所以您的一种方法是直接使用 RedistTemplate
,这样的方法对您有用。
@Service
public class TestService {
@Autowired
private RedisTemplate redisTemplate;
public void saveIt(TestObject testObject){
ValueOperations<ObjectId, TestObject> values = redisTemplate.opsForValue();
values.set(testObject.testId, testObject);
}
}
所以上面的代码将使用您的配置并使用 Jackson 作为键和值的映射器在 Redis 中生成字符串对。
但是,如果你想通过 CrudRepository
使用 Redis 存储库,你需要为 ObjectId
创建读写转换器,从和到 String
和 byte[]
并注册它们作为自定义 Redis 转换。
所以让我们为 ObjectId
<-> String
创建读写转换器
Reader
@Component
@ReadingConverter
@Slf4j
public class RedisReadingStringConverter implements Converter<String, ObjectId> {
private ObjectMapper objectMapper = new ObjectMapper();
@Override
public ObjectId convert(String source) {
try {
return objectMapper.readValue(source, ObjectId.class);
} catch (IOException e) {
log.warn("Error while converting to ObjectId.", e);
throw new IllegalArgumentException("Can not convert to ObjectId");
}
}
}
作家
@Component
@WritingConverter
@Slf4j
public class RedisWritingStringConverter implements Converter<ObjectId, String> {
private ObjectMapper objectMapper = new ObjectMapper();
@Override
public String convert(ObjectId source) {
try {
return objectMapper.writeValueAsString(source);
} catch (JsonProcessingException e) {
log.warn("Error while converting ObjectId to String.", e);
throw new IllegalArgumentException("Can not convert ObjectId to String");
}
}
}
以及ObjectId <-> byte[]的读写转换器
作家
@Component
@WritingConverter
public class RedisWritingByteConverter implements Converter<ObjectId, byte[]> {
Jackson2JsonRedisSerializer<ObjectId> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(ObjectId.class);
@Override
public byte[] convert(ObjectId source) {
return jackson2JsonRedisSerializer.serialize(source);
}
}
Reader
@Component
@ReadingConverter
public class RedisReadingByteConverter implements Converter<byte[], ObjectId> {
Jackson2JsonRedisSerializer<ObjectId> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(ObjectId.class);
@Override
public ObjectId convert(byte[] source) {
return jackson2JsonRedisSerializer.deserialize(source);
}
}
最后添加 Redis 自定义对话。只需将代码放入 RedisConfiguration
@Bean
public RedisCustomConversions redisCustomConversions(RedisReadingByteConverter readingConverter,
RedisWritingByteConverter redisWritingConverter,
RedisWritingStringConverter redisWritingByteConverter,
RedisReadingStringConverter redisReadingByteConverter) {
return new RedisCustomConversions(Arrays.asList(readingConverter, redisWritingConverter, redisWritingByteConverter, redisReadingByteConverter));
}
现在,在创建转换器并将其注册为自定义 Redis 转换器后,RedisKeyValueTemplate
可以使用它们,您的代码应该可以按预期工作。
我们尝试在我们的项目中使用 Spring 数据 CrudRepository
来为我们的域对象提供持久性。
一开始,我选择了 REDIS 作为后端,因为在第一次使用 CrudRepository<ExperimentDomainObject, String>
的实验中,得到它 运行 很容易。
当试图将它放入我们的生产代码中时,事情变得更加复杂,因为这里我们的域对象没有必要使用简单类型作为键,所以存储库是 CrudRepository<TestObject, ObjectId>
。
现在我得到了异常:
No converter found capable of converting from type [...ObjectId] to type [byte[]]
搜索此异常,RedisTemplate
配置进行了未受过教育的试验。 (对于我的实验,我使用的是 emdedded-redis)
我的想法是,提供 RedisTemplate<Object, Object>
而不是 RedisTemplate<String, Object>
以允许使用 Jackson2JsonRedisSerializer
来完成 keySerializer 的工作。
仍然调用 testRepository.save(testObject)
失败。
请看我的代码:
我有 public 个字段,为了本示例的简洁,省略了导入。如果需要它们(使其成为 MVCE),我会很乐意提供它们。请给我留言。
dependencies:
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-data-redis'
implementation 'org.springframework.boot:spring-boot-starter-web'
compileOnly 'org.projectlombok:lombok'
annotationProcessor 'org.projectlombok:lombok'
implementation group: 'redis.clients', name: "jedis", version: '2.9.0'
implementation group: 'it.ozimov', name: 'embedded-redis', version: '0.7.2'
}
RedisConfiguration:
@Configuration
@EnableRedisRepositories
public class RedisConfiguration {
@Bean
JedisConnectionFactory jedisConnectionFactory() {
return new JedisConnectionFactory();
}
@Bean
public RedisTemplate<Object, Object> redisTemplate() {
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
final RedisTemplate<Object, Object> template = new RedisTemplate<>();
template.setConnectionFactory(jedisConnectionFactory());
template.setDefaultSerializer(jackson2JsonRedisSerializer);
template.setKeySerializer(jackson2JsonRedisSerializer);
template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.setValueSerializer(jackson2JsonRedisSerializer);
template.setEnableDefaultSerializer(true);
return template;
}
}
TestObject
@RedisHash("test")
public class TestObject
{
@Id public ObjectId testId;
public String value;
public TestObject(ObjectId id, String value)
{
this.testId = id;
this.value = value; // In experiment this is: "magic"
}
}
ObjectId
@EqualsAndHashCode
public class ObjectId {
public String creator; // In experiment, this is "me"
public String name; // In experiment, this is "fool"
}
TestRepository
@Repository
public interface TestRepository extends CrudRepository<TestObject, ObjectId>
{
}
EmbeddedRedisConfiguration
@Configuration
public class EmbeddedRedisConfiguration
{
private final redis.embedded.RedisServer redisServer;
EmbeddedRedisConfiguration(RedisProperties redisProperties)
{
this.redisServer = new redis.embedded.RedisServer(redisProperties.getPort());
}
@PostConstruct
public void init()
{
redisServer.start();
}
@PreDestroy
public void shutdown()
{
redisServer.stop();
}
}
Application:
@SpringBootApplication
public class ExperimentApplication
{
public static void main(String[] args)
{
SpringApplication.run(ExperimentApplication.class, args);
}
}
不是想要的答案:
当然,我可能会引入一些特殊的ID,它是一种简单的数据类型,例如我使用 jacksons ObjectMapper 手动构建的 JSON-String,然后使用 CrudRepository<TestObject, String>
.
同时我也试过:
RedisTemplate<String, String>
RedisTemplate<String, Object>
- 自动装配
RedisTemplate
并设置其默认序列化器 - 正在注册
Converter<ObjectId, byte[]>
- 自动装配
ConverterRegistry
- 自动装配
GenericConversionService
但显然他们错了。
- 自动装配
基本上,Redis 存储库在底层使用 RedisKeyValueTemplate
将数据存储为键 (Id) 和值对。所以你配置的 RedisTemplate
除非你直接使用它,否则不会起作用。
所以您的一种方法是直接使用 RedistTemplate
,这样的方法对您有用。
@Service
public class TestService {
@Autowired
private RedisTemplate redisTemplate;
public void saveIt(TestObject testObject){
ValueOperations<ObjectId, TestObject> values = redisTemplate.opsForValue();
values.set(testObject.testId, testObject);
}
}
所以上面的代码将使用您的配置并使用 Jackson 作为键和值的映射器在 Redis 中生成字符串对。
但是,如果你想通过 CrudRepository
使用 Redis 存储库,你需要为 ObjectId
创建读写转换器,从和到 String
和 byte[]
并注册它们作为自定义 Redis 转换。
所以让我们为 ObjectId
<-> String
Reader
@Component
@ReadingConverter
@Slf4j
public class RedisReadingStringConverter implements Converter<String, ObjectId> {
private ObjectMapper objectMapper = new ObjectMapper();
@Override
public ObjectId convert(String source) {
try {
return objectMapper.readValue(source, ObjectId.class);
} catch (IOException e) {
log.warn("Error while converting to ObjectId.", e);
throw new IllegalArgumentException("Can not convert to ObjectId");
}
}
}
作家
@Component
@WritingConverter
@Slf4j
public class RedisWritingStringConverter implements Converter<ObjectId, String> {
private ObjectMapper objectMapper = new ObjectMapper();
@Override
public String convert(ObjectId source) {
try {
return objectMapper.writeValueAsString(source);
} catch (JsonProcessingException e) {
log.warn("Error while converting ObjectId to String.", e);
throw new IllegalArgumentException("Can not convert ObjectId to String");
}
}
}
以及ObjectId <-> byte[]的读写转换器
作家
@Component
@WritingConverter
public class RedisWritingByteConverter implements Converter<ObjectId, byte[]> {
Jackson2JsonRedisSerializer<ObjectId> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(ObjectId.class);
@Override
public byte[] convert(ObjectId source) {
return jackson2JsonRedisSerializer.serialize(source);
}
}
Reader
@Component
@ReadingConverter
public class RedisReadingByteConverter implements Converter<byte[], ObjectId> {
Jackson2JsonRedisSerializer<ObjectId> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(ObjectId.class);
@Override
public ObjectId convert(byte[] source) {
return jackson2JsonRedisSerializer.deserialize(source);
}
}
最后添加 Redis 自定义对话。只需将代码放入 RedisConfiguration
@Bean
public RedisCustomConversions redisCustomConversions(RedisReadingByteConverter readingConverter,
RedisWritingByteConverter redisWritingConverter,
RedisWritingStringConverter redisWritingByteConverter,
RedisReadingStringConverter redisReadingByteConverter) {
return new RedisCustomConversions(Arrays.asList(readingConverter, redisWritingConverter, redisWritingByteConverter, redisReadingByteConverter));
}
现在,在创建转换器并将其注册为自定义 Redis 转换器后,RedisKeyValueTemplate
可以使用它们,您的代码应该可以按预期工作。