Spring RedisTemplate : Serialise multiple Model cl

2019-02-04 08:06发布

I am using Spring Redis support to save my objects in Redis.

I have several DAOs which handle different Model classes :

eg : 'ShopperHistoryDao' which will save/retrieve objects of 'ShopperHistoryModel' 'ShopperItemHistoryDao' which will handle 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>

3条回答
The star\"
2楼-- · 2019-02-04 08:31

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.

查看更多
混吃等死
3楼-- · 2019-02-04 08:32

A bit of old thread, but you can do something like this:

<bean id="RedisTemplate" class="org.springframework.data.redis.core.RedisTemplate">
    <property name="connectionFactory" ref="jedisConnectionFactory" />
    <property name="valueSerializer">
        <bean id="jackson2JsonRedisSerializer" 
                        class="org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer">
            <constructor-arg type="java.lang.Class" value="Object.class" />
        </bean>   
    </property>
</bean>

Then in your Java class

@Autowire
private RedisTemplate redisTemplate;

public void save(Model model) {
    ObjectMapper obmap = new ObjectMapper();
    redisTemplate.opsForHash().putAll(mode.getId(), obmap.convertValue(model, Map.class));
}
查看更多
贼婆χ
4楼-- · 2019-02-04 08:53

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

查看更多
登录 后发表回答