I am using Spring Redis support to save my objects in Redis.
I have several DAOs which handle different Model classes:
For example, ShopperHistoryDao
save/retrieve objects of ShopperHistoryModel
, ShopperItemHistoryDao
save/retrieve objects of ItemHistoryModel
.
I want to use JacksonJsonRedisSerializer
to serialise/deserialize my objects to/from json.
But in the constructor of JacksonJsonRedisSerializer, it takes one specific Model class.
JacksonJsonRedisSerializer(Class<T> type)
Does that mean, I have to configure separate RedisTemplates
for each different Model class and use them in appropriate DAO implementation?
Something like:
<bean id="redisTemplateForShopperHistoryModel" class="org.springframework.data.redis.core.RedisTemplate">
<property name="connectionFactory" ref="jedisConnectionFactory" />
<property name="valueSerializer">
<bean id="redisJsonSerializer"
class="org.springframework.data.redis.serializer.JacksonJsonRedisSerializer">
<constructor-arg type="java.lang.Class" value="ShopperHistoryModel.class"/>
</bean>
</property>
</bean>
<bean id="redisTemplateForItemHistoryModel" class="org.springframework.data.redis.core.RedisTemplate">
<property name="connectionFactory" ref="jedisConnectionFactory" />
<property name="valueSerializer">
<bean id="redisJsonSerializer"
class="org.springframework.data.redis.serializer.JacksonJsonRedisSerializer">
<constructor-arg type="java.lang.Class" value="ItemHistoryModel.class"/>
</bean>
</property>
</bean>
GenericJackson2JsonRedisSerializer should do the job
@Bean
public RedisTemplate<String, Object> redisTemplate() {
RedisTemplate<String, Object> redisTemplate = new RedisTemplate<String, Object>();
redisTemplate.setConnectionFactory(jedisConnectionFactory());
redisTemplate.setKeySerializer(new StringRedisSerializer());
redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer());
return redisTemplate;
}
This will add @Class property to the JSON to understand the type, which helps Jackson to deserialize, so no need to explicitly map the model on the configuration class.
"{\"@class\":\"com.prnv.model.WhitePaper\",\"title\":\"Hey\",\"author\":{\"@class\":\"com.prnv.model.Author\",\"name\":\"Hello\"},\"description\":\"Description\"}"
In the service you can cache the model using
@Cacheable(value = "whitePaper", key = "#title")
public WhitePaper findWhitePaperByTitle(String title)
{
WhitePaper whitePaper = repository.findByTitle(title);
return whitePaper;
}
Check this article: http://blog.pranavek.com/2016/12/25/integrating-redis-with-spring-application
Yes, the RedisTemplate
seems to be designed to have a single instance of the value serializer.
I was going to suggest the possible workaround of having a RedisSerializer
which contains a Map of inner serializers so you can use one RedisTemplate
with a serializer that can handle multiple types - but since RedisSerializer
does not offer methods like boolean canDeserialize(..)
(as the HTTP MessageConverters in Spring MVC have) this doesn't seem possible.
So it seems that you are stuck with having multiple RedisTemplate
instances.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With