Spring Boot使用redis作數據緩存

1 添加redis支持

在pom.xml中添加html

Xml代碼   收藏代碼
  1. <dependency>  
  2.           <groupId>org.springframework.boot</groupId>  
  3.           <artifactId>spring-boot-starter-redis</artifactId>  
  4.       </dependency>  

 

2 redis配置

Java代碼   收藏代碼
  1. package com.wisely.ij.config;  
  2.   
  3. import com.fasterxml.jackson.annotation.JsonAutoDetect;  
  4. import com.fasterxml.jackson.annotation.PropertyAccessor;  
  5. import com.fasterxml.jackson.databind.ObjectMapper;  
  6. import org.springframework.cache.CacheManager;  
  7. import org.springframework.cache.annotation.CachingConfigurerSupport;  
  8. import org.springframework.cache.annotation.EnableCaching;  
  9. import org.springframework.cache.interceptor.KeyGenerator;  
  10. import org.springframework.context.annotation.Bean;  
  11. import org.springframework.context.annotation.Configuration;  
  12. import org.springframework.data.redis.cache.RedisCacheManager;  
  13. import org.springframework.data.redis.connection.RedisConnectionFactory;  
  14. import org.springframework.data.redis.core.RedisTemplate;  
  15. import org.springframework.data.redis.core.StringRedisTemplate;  
  16. import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;  
  17.   
  18. import java.lang.reflect.Method;  
  19.   
  20.   
  21. @Configuration  
  22. @EnableCaching  
  23. public class RedisConfig extends CachingConfigurerSupport{  
  24.   
  25.     @Bean  
  26.     public KeyGenerator wiselyKeyGenerator(){  
  27.         return new KeyGenerator() {  
  28.             @Override  
  29.             public Object generate(Object target, Method method, Object... params) {  
  30.                 StringBuilder sb = new StringBuilder();  
  31.                 sb.append(target.getClass().getName());  
  32.                 sb.append(method.getName());  
  33.                 for (Object obj : params) {  
  34.                     sb.append(obj.toString());  
  35.                 }  
  36.                 return sb.toString();  
  37.             }  
  38.         };  
  39.   
  40.     }  
  41.   
  42.     @Bean  
  43.     public CacheManager cacheManager(  
  44.             @SuppressWarnings("rawtypes") RedisTemplate redisTemplate) {  
  45.         return new RedisCacheManager(redisTemplate);  
  46.     }  
  47.   
  48.     @Bean  
  49.     public RedisTemplate<String, String> redisTemplate(  
  50.             RedisConnectionFactory factory) {  
  51.         StringRedisTemplate template = new StringRedisTemplate(factory);  
  52.         Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);  
  53.         ObjectMapper om = new ObjectMapper();  
  54.         om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);  
  55.         om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);  
  56.         jackson2JsonRedisSerializer.setObjectMapper(om);  
  57.         template.setValueSerializer(jackson2JsonRedisSerializer);  
  58.         template.afterPropertiesSet();  
  59.         return template;  
  60.     }  
  61. }  

 

 

3 redis服務器配置

Properties代碼   收藏代碼
  1. # REDIS (RedisProperties)  
  2. spring.redis.database= # database name  
  3. spring.redis.host=localhost # server host  
  4. spring.redis.password= # server password  
  5. spring.redis.port=6379 # connection port  
  6. spring.redis.pool.max-idle=8 # pool settings ...  
  7. spring.redis.pool.min-idle=0  
  8. spring.redis.pool.max-active=8  
  9. spring.redis.pool.max-wait=-1  
  10. spring.redis.sentinel.master= # name of Redis server  
  11. spring.redis.sentinel.nodes= # comma-separated list of host:port pairs  

 

4 應用

測試兩個實體類java

Java代碼   收藏代碼
  1. package com.wisely.ij.domain;  
  2.   
  3.   
  4. public class Address {  
  5.     private Long id;  
  6.     private String province;  
  7.     private String city;  
  8.   
  9.     public Address(Long id,String province, String city) {  
  10.         this.id = id;  
  11.         this.province = province;  
  12.         this.city = city;  
  13.     }  
  14.   
  15.     public Address() {  
  16.     }  
  17.   
  18.     public Long getId() {  
  19.         return id;  
  20.     }  
  21.   
  22.     public void setId(Long id) {  
  23.         this.id = id;  
  24.     }  
  25.   
  26.     public String getProvince() {  
  27.         return province;  
  28.     }  
  29.   
  30.     public void setProvince(String province) {  
  31.         this.province = province;  
  32.     }  
  33.   
  34.     public String getCity() {  
  35.         return city;  
  36.     }  
  37.   
  38.     public void setCity(String city) {  
  39.         this.city = city;  
  40.     }  
  41. }  

 

Java代碼   收藏代碼
  1. package com.wisely.ij.domain;  
  2.   
  3.   
  4. public class User {  
  5.     private Long id;  
  6.     private String firstName;  
  7.     private String lastName;  
  8.   
  9.     public User(Long id,String firstName, String lastName) {  
  10.         this.id = id ;  
  11.         this.firstName = firstName;  
  12.         this.lastName = lastName;  
  13.     }  
  14.   
  15.     public User() {  
  16.     }  
  17.   
  18.     public Long getId() {  
  19.         return id;  
  20.     }  
  21.   
  22.     public void setId(Long id) {  
  23.         this.id = id;  
  24.     }  
  25.   
  26.     public String getFirstName() {  
  27.         return firstName;  
  28.     }  
  29.   
  30.     public void setFirstName(String firstName) {  
  31.         this.firstName = firstName;  
  32.     }  
  33.   
  34.     public String getLastName() {  
  35.         return lastName;  
  36.     }  
  37.   
  38.     public void setLastName(String lastName) {  
  39.         this.lastName = lastName;  
  40.     }  
  41. }  

 

 使用演示node

Java代碼   收藏代碼
  1. package com.wisely.ij.service;  
  2.   
  3. import com.wisely.ij.domain.Address;  
  4. import com.wisely.ij.domain.User;  
  5. import org.springframework.cache.annotation.Cacheable;  
  6. import org.springframework.stereotype.Service;  
  7.   
  8. /** 
  9.  * Created by wisely on 2015/5/25. 
  10.  */  
  11. @Service  
  12. public class DemoService {  
  13.     @Cacheable(value = "usercache",keyGenerator = "wiselyKeyGenerator")  
  14.     public User findUser(Long id,String firstName,String lastName){  
  15.         System.out.println("無緩存的時候調用這裏");  
  16.         return new User(id,firstName,lastName);  
  17.     }  
  18.     @Cacheable(value = "addresscache",keyGenerator = "wiselyKeyGenerator")  
  19.     public Address findAddress(Long id,String province,String city){  
  20.         System.out.println("無緩存的時候調用這裏");  
  21.         return new Address(id,province,city);  
  22.     }  
  23. }  

 

Java代碼   收藏代碼
  1. package com.wisely.ij.web;  
  2.   
  3. import com.wisely.ij.domain.Address;  
  4. import com.wisely.ij.domain.User;  
  5. import com.wisely.ij.service.DemoService;  
  6. import org.springframework.beans.factory.annotation.Autowired;  
  7. import org.springframework.stereotype.Controller;  
  8. import org.springframework.web.bind.annotation.RequestMapping;  
  9. import org.springframework.web.bind.annotation.ResponseBody;  
  10.   
  11. /** 
  12.  * Created by wisely on 2015/5/25. 
  13.  */  
  14.   
  15. @Controller  
  16. public class DemoController {  
  17.   
  18.     @Autowired  
  19.     DemoService demoService;  
  20.   
  21.     @RequestMapping("/test")  
  22.     @ResponseBody  
  23.     public String putCache(){  
  24.         demoService.findUser(1l,"wang","yunfei");  
  25.         demoService.findAddress(1l,"anhui","hefei");  
  26.         System.out.println("若下面沒出現「無緩存的時候調用」字樣且能打印出數據表示測試成功");  
  27.         return "ok";  
  28.     }  
  29.     @RequestMapping("/test2")  
  30.     @ResponseBody  
  31.     public String testCache(){  
  32.         User user = demoService.findUser(1l,"wang","yunfei");  
  33.         Address address =demoService.findAddress(1l,"anhui","hefei");  
  34.         System.out.println("我這裏沒執行查詢");  
  35.         System.out.println("user:"+"/"+user.getFirstName()+"/"+user.getLastName());  
  36.         System.out.println("address:"+"/"+address.getProvince()+"/"+address.getCity());  
  37.         return "ok";  
  38.     }  
  39. }  

 

5 檢驗

 

先訪問http://localhost:8080/test 保存緩存git



 

再訪問http://localhost:8080/test2 調用緩存裏的數據github

 



 

 http://wiselyman.iteye.com/blog/2184884web

整合 spring 4(包括mvc、context、orm) + mybatis 3 示例》一文簡要介紹了最新版本的 Spring MVC、IOC、MyBatis ORM 三者的整合以及聲明式事務處理。如今咱們須要把緩存也整合進來,緩存咱們選用的是 Redis,本文將在該文示例基礎上介紹 Redis 緩存 + Spring 的集成。關於 Redis 服務器的搭建請參考博客《Redhat5.8 環境下編譯安裝 Redis 並將其註冊爲系統服務》。redis

1. 依賴包安裝

pom.xml 加入:spring

[html]  view plain  copy
 
 print?
  1. <!-- redis cache related.....start -->  
  2. <dependency>  
  3.     <groupId>org.springframework.data</groupId>  
  4.     <artifactId>spring-data-redis</artifactId>  
  5.     <version>1.6.0.RELEASE</version>  
  6. </dependency>  
  7. <dependency>  
  8.     <groupId>redis.clients</groupId>  
  9.     <artifactId>jedis</artifactId>  
  10.     <version>2.7.3</version>  
  11. </dependency>  
  12. <!-- redis cache related.....end -->  

 

2. Spring 項目集成進緩存支持

要啓用緩存支持,咱們須要建立一個新的 CacheManager bean。CacheManager 接口有不少實現,本文演示的是和 Redis 的集成,天然就是用 RedisCacheManager 了。Redis 不是應用的共享內存,它只是一個內存服務器,就像 MySql 似的,咱們須要將應用鏈接到它並使用某種「語言」進行交互,所以咱們還須要一個鏈接工廠以及一個 Spring 和 Redis 對話要用的 RedisTemplate,這些都是 Redis 緩存所必需的配置,把它們都放在自定義的 CachingConfigurerSupport 中:數據庫

[java]  view plain  copy
 
 print?
  1. /** 
  2.  * File Name:RedisCacheConfig.java 
  3.  * 
  4.  * Copyright Defonds Corporation 2015  
  5.  * All Rights Reserved 
  6.  * 
  7.  */  
  8. package com.defonds.bdp.cache.redis;  
  9.   
  10. import org.springframework.cache.CacheManager;  
  11. import org.springframework.cache.annotation.CachingConfigurerSupport;  
  12. import org.springframework.cache.annotation.EnableCaching;  
  13. import org.springframework.context.annotation.Bean;  
  14. import org.springframework.context.annotation.Configuration;  
  15. import org.springframework.data.redis.cache.RedisCacheManager;  
  16. import org.springframework.data.redis.connection.RedisConnectionFactory;  
  17. import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;  
  18. import org.springframework.data.redis.core.RedisTemplate;  
  19.   
  20. /** 
  21.  *  
  22.  * Project Name:bdp  
  23.  * Type Name:RedisCacheConfig  
  24.  * Type Description: 
  25.  *  Author:Defonds 
  26.  * Create Date:2015-09-21 
  27.  *  
  28.  * @version 
  29.  *  
  30.  */  
  31. @Configuration  
  32. @EnableCaching  
  33. public class RedisCacheConfig extends CachingConfigurerSupport {  
  34.   
  35.     @Bean  
  36.     public JedisConnectionFactory redisConnectionFactory() {  
  37.         JedisConnectionFactory redisConnectionFactory = new JedisConnectionFactory();  
  38.   
  39.         // Defaults  
  40.         redisConnectionFactory.setHostName("192.168.1.166");  
  41.         redisConnectionFactory.setPort(6379);  
  42.         return redisConnectionFactory;  
  43.     }  
  44.   
  45.     @Bean  
  46.     public RedisTemplate<String, String> redisTemplate(RedisConnectionFactory cf) {  
  47.         RedisTemplate<String, String> redisTemplate = new RedisTemplate<String, String>();  
  48.         redisTemplate.setConnectionFactory(cf);  
  49.         return redisTemplate;  
  50.     }  
  51.   
  52.     @Bean  
  53.     public CacheManager cacheManager(RedisTemplate redisTemplate) {  
  54.         RedisCacheManager cacheManager = new RedisCacheManager(redisTemplate);  
  55.   
  56.         // Number of seconds before expiration. Defaults to unlimited (0)  
  57.         cacheManager.setDefaultExpiration(3000); // Sets the default expire time (in seconds)  
  58.         return cacheManager;  
  59.     }  
  60.       
  61. }  


固然也別忘了把這些 bean 注入 Spring,否則配置無效。在 applicationContext.xml 中加入如下:json

[html]  view plain  copy
 
 print?
  1. <context:component-scan base-package="com.defonds.bdp.cache.redis" />  

 

3. 緩存某些方法的執行結果

設置好緩存配置以後咱們就可使用 @Cacheable 註解來緩存方法執行的結果了,好比根據省份名檢索城市的 provinceCities 方法和根據 city_code 檢索城市的 searchCity 方法:

[java]  view plain  copy
 
 print?
  1. // R  
  2. @Cacheable("provinceCities")  
  3. public List<City> provinceCities(String province) {  
  4.     logger.debug("province=" + province);  
  5.     return this.cityMapper.provinceCities(province);  
  6. }  
  7.   
  8. // R  
  9. @Cacheable("searchCity")  
  10. public City searchCity(String city_code){  
  11.     logger.debug("city_code=" + city_code);  
  12.     return this.cityMapper.searchCity(city_code);     
  13. }  

 

4. 緩存數據一致性保證

CRUD (Create 建立,Retrieve 讀取,Update 更新,Delete 刪除) 操做中,除了 R 具有冪等性,其餘三個發生的時候均可能會形成緩存結果和數據庫不一致。爲了保證緩存數據的一致性,在進行 CUD 操做的時候咱們須要對可能影響到的緩存進行更新或者清除。

[java]  view plain  copy
 
 print?
  1. // C  
  2. @CacheEvict(value = { "provinceCities"}, allEntries = true)  
  3. public void insertCity(String city_code, String city_jb,   
  4.         String province_code, String city_name,  
  5.         String city, String province) {  
  6.     City cityBean = new City();  
  7.     cityBean.setCityCode(city_code);  
  8.     cityBean.setCityJb(city_jb);  
  9.     cityBean.setProvinceCode(province_code);  
  10.     cityBean.setCityName(city_name);  
  11.     cityBean.setCity(city);  
  12.     cityBean.setProvince(province);  
  13.     this.cityMapper.insertCity(cityBean);  
  14. }  
  15. // U  
  16. @CacheEvict(value = { "provinceCities", "searchCity" }, allEntries = true)  
  17. public int renameCity(String city_code, String city_name) {  
  18.     City city = new City();  
  19.     city.setCityCode(city_code);  
  20.     city.setCityName(city_name);  
  21.     this.cityMapper.renameCity(city);  
  22.     return 1;  
  23. }  
  24.   
  25. // D  
  26. @CacheEvict(value = { "provinceCities", "searchCity" }, allEntries = true)  
  27. public int deleteCity(String city_code) {  
  28.     this.cityMapper.deleteCity(city_code);  
  29.     return 1;  
  30. }  


業務考慮,本示例用的都是 @CacheEvict 清除緩存。若是你的 CUD 可以返回 City 實例,也可使用 @CachePut 更新緩存策略。筆者推薦能用 @CachePut 的地方就不要用 @CacheEvict,由於後者將全部相關方法的緩存都清理掉,好比上面三個方法中的任意一個被調用了的話,provinceCities 方法的全部緩存將被清除。

5. 自定義緩存數據 key 生成策略

對於使用 @Cacheable 註解的方法,每一個緩存的 key 生成策略默認使用的是參數名+參數值,好比如下方法:

[java]  view plain  copy
 
 print?
  1. @Cacheable("users")  
  2. public User findByUsername(String username)  


這個方法的緩存將保存於 key 爲 users~keys 的緩存下,對於 username 取值爲 "趙德芳" 的緩存,key 爲 "username-趙德芳"。通常狀況下沒啥問題,二般狀況如方法 key 取值相等而後參數名也同樣的時候就出問題了,如:

[java]  view plain  copy
 
 print?
  1. @Cacheable("users")  
  2. public Integer getLoginCountByUsername(String username)  


這個方法的緩存也將保存於 key 爲 users~keys 的緩存下。對於 username 取值爲 "趙德芳" 的緩存,key 也爲 "username-趙德芳",將另一個方法的緩存覆蓋掉。
解決辦法是使用自定義緩存策略,對於同一業務(同一業務邏輯處理的方法,哪怕是集羣/分佈式系統),生成的 key 始終一致,對於不一樣業務則不一致:

[java]  view plain  copy
 
 print?
  1. @Bean  
  2. public KeyGenerator customKeyGenerator() {  
  3.     return new KeyGenerator() {  
  4.         @Override  
  5.         public Object generate(Object o, Method method, Object... objects) {  
  6.             StringBuilder sb = new StringBuilder();  
  7.             sb.append(o.getClass().getName());  
  8.             sb.append(method.getName());  
  9.             for (Object obj : objects) {  
  10.                 sb.append(obj.toString());  
  11.             }  
  12.             return sb.toString();  
  13.         }  
  14.     };  
  15. }  


因而上述兩個方法,對於 username 取值爲 "趙德芳" 的緩存,雖然都仍是存放在 key 爲 users~keys 的緩存下,但因爲 key 分別爲 "類名-findByUsername-username-趙德芳" 和 "類名-getLoginCountByUsername-username-趙德芳",因此也不會有問題。
這對於集羣系統、分佈式系統之間共享緩存很重要,真正實現了分佈式緩存。
筆者建議:緩存方法的 @Cacheable 最好使用方法名,避免不一樣的方法的 @Cacheable 值一致,而後再配以以上緩存策略。

6. 緩存的驗證

6.1 緩存的驗證

爲了肯定每一個緩存方法到底有沒有走緩存,咱們打開了 MyBatis 的 SQL 日誌輸出,而且爲了演示清楚,咱們還清空了測試用 Redis 數據庫。
先來驗證 provinceCities 方法緩存,Eclipse 啓動 tomcat 加載項目完畢,使用 JMeter 調用 /bdp/city/province/cities.json 接口:
使用 JMeter 調用 /bdp/city/province/cities.json 接口.png
Eclipse 控制檯輸出以下:
Eclipse 控制檯輸出以下.png
說明這一次請求沒有命中緩存,走的是 db 查詢。JMeter 再次請求,Eclipse 控制檯輸出:
Eclipse 控制檯輸出
標紅部分如下是這一次請求的 log,沒有訪問 db 的 log,緩存命中。查看本次請求的 Redis 存儲狀況:
查看本次請求的 Redis 存儲狀況.png
一樣能夠驗證 city_code 爲 1492 的 searchCity 方法的緩存是否有效:
一樣能夠驗證 city_code 爲 1492 的 searchCity 方法的緩存是否有效.png
圖中標紅部分是 searchCity 的緩存存儲狀況。

6.2 緩存一致性的驗證

先來驗證 insertCity 方法的緩存配置,JMeter 調用 /bdp/city/create.json 接口:
JMeter 調用 /bdp/city/create.json 接口.png
以後看 Redis 存儲:
以後看 Redis 存儲
能夠看出 provinceCities 方法的緩存已被清理掉,insertCity 方法的緩存奏效。
而後驗證 renameCity 方法的緩存配置,JMeter 調用 /bdp/city/rename.json 接口:
JMeter 調用 /bdp/city/rename.json 接口.png
以後再看 Redis 存儲:
以後再看 Redis 存儲.png
searchCity 方法的緩存也已被清理,renameCity 方法的緩存也奏效。

7. 注意事項

  1. 要緩存的 Java 對象必須實現 Serializable 接口,由於 Spring 會將對象先序列化再存入 Redis,好比本文中的 com.defonds.bdp.city.bean.City 類,若是不實現 Serializable 的話將會遇到相似這種錯誤:nested exception is java.lang.IllegalArgumentException: DefaultSerializer requires a Serializable payload but received an object of type [com.defonds.bdp.city.bean.City]]。
  2. 緩存的生命週期咱們能夠配置,而後託管 Spring CacheManager,不要試圖經過 redis-cli 命令行去管理緩存。好比 provinceCities 方法的緩存,某個省份的查詢結果會被以 key-value 的形式存放在 Redis,key 就是咱們剛纔自定義生成的 key,value 是序列化後的對象,這個 key 會被放在 key 名爲 provinceCities~keys key-value 存儲中,參考下圖"provinceCities 方法在 Redis 中的緩存狀況"。能夠經過 redis-cli 使用 del 命令將 provinceCities~keys 刪除,但每一個省份的緩存卻不會被清除。
  3. CacheManager 必須設置緩存過時時間,不然緩存對象將永不過時,這樣作的緣由如上,避免一些野數據「永久保存」。此外,設置緩存過時時間也有助於資源利用最大化,由於緩存裏保留的永遠是熱點數據。
  4. 緩存適用於讀多寫少的場合,查詢時緩存命中率很低、寫操做很頻繁等場景不適宜用緩存。

provinceCities方法在Redis中的存儲.png

後記

本文完整 Eclipse 下的開發項目示例已上傳 CSDN 資源,有興趣的朋友能夠去下載下來參考:http://download.csdn.net/detail/defonds/9137505

參考資料

http://blog.csdn.net/defonds/article/details/48716161

本文介紹瞭如何使用註解的方式,將Redis緩存整合到你的Spring項目。

首先咱們將使用jedis驅動,進而開始配置咱們的Gradle。

 

group 'com.gkatzioura.spring'
version '1.0-SNAPSHOT'
apply plugin: 'java'
apply plugin: 'eclipse'
apply plugin: 'idea'
apply plugin: 'spring-boot'
buildscript {
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath("org.springframework.boot:spring-boot-gradle-plugin:1.2.5.RELEASE")
    }
}
jar {
    baseName = 'gs-serving-web-content'
    version =  '0.1.0'
}
sourceCompatibility = 1.8
repositories {
    mavenCentral()
}
dependencies {
    compile "org.springframework.boot:spring-boot-starter-thymeleaf"
    compile 'org.slf4j:slf4j-api:1.6.6'
    compile 'ch.qos.logback:logback-classic:1.0.13'
    compile 'redis.clients:jedis:2.7.0'
    compile 'org.springframework.data:spring-data-redis:1.5.0.RELEASE'
    testCompile group: 'junit', name: 'junit', version: '4.11'
}
task wrapper(type: Wrapper) {
    gradleVersion = '2.3'
}

緊接着咱們將使用Spring註解,繼續執行Redis裝載配置。

package com.gkatzioura.spring.config;
import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.CachingConfigurerSupport;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.RedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;
@Configuration
@EnableCaching
public class RedisConfig extends CachingConfigurerSupport {
    @Bean
    public JedisConnectionFactory redisConnectionFactory() {
        JedisConnectionFactory jedisConnectionFactory = new JedisConnectionFactory();
        jedisConnectionFactory.setUsePool(true);
        return jedisConnectionFactory;
    }
    @Bean
    public RedisSerializer redisStringSerializer() {
        StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
        return stringRedisSerializer;
    }
    @Bean(name="redisTemplate")
    public RedisTemplate<String, String> redisTemplate(RedisConnectionFactory cf,RedisSerializer redisSerializer) {
        RedisTemplate<String, String> redisTemplate = new RedisTemplate<String, String>();
        redisTemplate.setConnectionFactory(cf);
        redisTemplate.setDefaultSerializer(redisSerializer);
        return redisTemplate;
    }
    @Bean
    public CacheManager cacheManager() {
        return new RedisCacheManager(redisTemplate(redisConnectionFactory(),redisStringSerializer()));
    }
}

 

下一步將建立緩存接口CacheService。

package com.gkatzioura.spring.cache;
import java.util.Date;
import java.util.List;
public interface CacheService {
    public void addMessage(String user,String message);
    public List<String> listMessages(String user);
}

固然用戶既能夠增長一條消息也能取回一條消息。所以,在實現過程當中,用戶相關信息的存在時間將默認設爲一分鐘。

咱們用Redis來繼承實現CacheService接口。

package com.gkatzioura.spring.cache.impl;
import com.gkatzioura.spring.cache.CacheService;
import org.springframework.data.redis.core.ListOperations;
import org.springframework.data.redis.core.RedisOperations;
import org.springframework.data.redis.core.SetOperations;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import java.time.ZonedDateTime;
import java.time.temporal.ChronoUnit;
import java.util.Date;
import java.util.List;
@Service("cacheService")
public class RedisService implements CacheService {
    @Resource(name = "redisTemplate")
    private ListOperations<String, String> messageList;
    @Resource(name = "redisTemplate")
    private RedisOperations<String,String> latestMessageExpiration;
    @Override
    public void addMessage(String user,String message) {
        messageList.leftPush(user,message);
        ZonedDateTime zonedDateTime = ZonedDateTime.now();
        Date date = Date.from(zonedDateTime.plus(1, ChronoUnit.MINUTES).toInstant());
        latestMessageExpiration.expireAt(user,date);
    }
    @Override
    public List<String> listMessages(String user) {
        return messageList.range(user,0,-1);
    }
}

咱們的緩存機制將保留每一個用戶發送的消息列表。爲了實現這個功能咱們將調用ListOperations接口,同時將每一個user做爲一個key鍵值。經過RedisOperations接口,咱們能夠爲key設置特定存在時長。在本例中,主要使用的是 user key。

下一步咱們將建立一個controller注入緩存服務。

package com.gkatzioura.spring.controller;
import com.gkatzioura.spring.cache.CacheService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import java.util.List;
@RestController
public class MessageController {
    @Autowired
    private CacheService cacheService;
    @RequestMapping(value = "/message",method = RequestMethod.GET)
    @ResponseBody
    public List<String> greeting(String user) {
        List<String> messages = cacheService.listMessages(user);
        return messages;
    }
    @RequestMapping(value = "/message",method = RequestMethod.POST)
    @ResponseBody
    public String saveGreeting(String user,String message) {
        cacheService.addMessage(user,message);
        return "OK";
    }
}

最後完成類Application的建立。

package com.gkatzioura.spring;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

通過如上步驟,接下來直接運行Application便可。

 

原文連接:Integrate Redis into a Spring Project( 譯者/丘志鵬 審校/朱正貴 責編/仲浩)

 

http://www.csdn.net/article/2015-09-01/2825600

使用Spring Cache + Redis + Jackson Serializer緩存數據庫查詢結果中序列化問題的解決

應用場景

咱們但願經過緩存來減小對關係型數據庫的查詢次數,減輕數據庫壓力。在執行DAO類的select***()query***()方法時,先從Redis中查詢有沒有緩存數據,若是有則直接從Redis拿到結果,若是沒有再向數據庫發起查詢請求取數據。

序列化問題

要把domain object作爲key-value對保存在redis中,就必需要解決對象的序列化問題。Spring Data Redis給咱們提供了一些現成的方案:

  • JdkSerializationRedisSerializer. 使用JDK提供的序列化功能。 優勢是反序列化時不須要提供類型信息(class),但缺點是序列化後的結果很是龐大,是JSON格式的5倍左右,這樣就會消耗redis服務器的大量內存。
  • Jackson2JsonRedisSerializer. 使用Jackson庫將對象序列化爲JSON字符串。優勢是速度快,序列化後的字符串短小精悍。
    但缺點也很是致命,那就是此類的構造函數中有一個類型參數,必須提供要序列化對象的類型信息(.class對象)。 經過查看源代碼,發現其只在反序列化過程當中用到了類型信息。

若是用方案一,就必須付出緩存多佔用4倍內存的代價,實在承受不起。若是用方案二,則必須給每一種domain對象都配置一個Serializer,即若是個人應用裏有100種domain對象,那就必須在spring配置文件中配置100個Jackson2JsonRedisSerializer,這顯然是不現實的。

經過google, 發現spring data redis項目中有一個#145 pull request, 而這個提交請求的內容正是解決Jackson必須提供類型信息的問題。然而不幸的是這個請求尚未被merge。但咱們能夠把代碼copy一下放到本身的項目中:

/** * @author Christoph Strobl * @since 1.6 */ public class GenericJackson2JsonRedisSerializer implements RedisSerializer<Object> { private final ObjectMapper mapper; /** * Creates {@link GenericJackson2JsonRedisSerializer} and configures {@link ObjectMapper} for default typing. */ public GenericJackson2JsonRedisSerializer() { this((String) null); } /** * Creates {@link GenericJackson2JsonRedisSerializer} and configures {@link ObjectMapper} for default typing using the * given {@literal name}. In case of an {@literal empty} or {@literal null} String the default * {@link JsonTypeInfo.Id#CLASS} will be used. * * @param classPropertyTypeName Name of the JSON property holding type information. Can be {@literal null}. */ public GenericJackson2JsonRedisSerializer(String classPropertyTypeName) { this(new ObjectMapper()); if (StringUtils.hasText(classPropertyTypeName)) { mapper.enableDefaultTypingAsProperty(DefaultTyping.NON_FINAL, classPropertyTypeName); } else { mapper.enableDefaultTyping(DefaultTyping.NON_FINAL, As.PROPERTY); } } /** * Setting a custom-configured {@link ObjectMapper} is one way to take further control of the JSON serialization * process. For example, an extended {@link SerializerFactory} can be configured that provides custom serializers for * specific types. * * @param mapper must not be {@literal null}. */ public GenericJackson2JsonRedisSerializer(ObjectMapper mapper) { Assert.notNull(mapper, "ObjectMapper must not be null!"); this.mapper = mapper; } /* * (non-Javadoc) * @see org.springframework.data.redis.serializer.RedisSerializer#serialize(java.lang.Object) */ @Override public byte[] serialize(Object source) throws SerializationException { if (source == null) { return SerializationUtils.EMPTY_ARRAY; } try { return mapper.writeValueAsBytes(source); } catch (JsonProcessingException e) { throw new SerializationException("Could not write JSON: " + e.getMessage(), e); } } /* * (non-Javadoc) * @see org.springframework.data.redis.serializer.RedisSerializer#deserialize(byte[]) */ @Override public Object deserialize(byte[] source) throws SerializationException { return deserialize(source, Object.class); } /** * @param source can be {@literal null}. * @param type must not be {@literal null}. * @return {@literal null} for empty source. * @throws SerializationException */ public <T> T deserialize(byte[] source, Class<T> type) throws SerializationException { Assert.notNull(type, "Deserialization type must not be null! Pleaes provide Object.class to make use of Jackson2 default typing."); if (SerializationUtils.isEmpty(source)) { return null; } try { return mapper.readValue(source, type); } catch (Exception ex) { throw new SerializationException("Could not read JSON: " + ex.getMessage(), ex); } } }

而後在配置文件中使用這個GenericJackson2JsonRedisSerializer:

<bean id="jacksonSerializer" class="com.fh.taolijie.component.GenericJackson2JsonRedisSerializer"> </bean>

從新構建部署,咱們發現這個serializer能夠同時支持多種不一樣類型的domain對象,問題解決。

http://www.myexception.cn/database/1958643.html

     spring-data-redis提供了多種serializer策略,這對使用jedis的開發者而言,實在是很是便捷。sdr提供了4種內置的serializer:

  • JdkSerializationRedisSerializer:使用JDK的序列化手段(serializable接口,ObjectInputStrean,ObjectOutputStream),數據以字節流存儲
  • StringRedisSerializer:字符串編碼,數據以string存儲
  • JacksonJsonRedisSerializer:json格式存儲
  • OxmSerializer:xml格式存儲

    其中JdkSerializationRedisSerializer和StringRedisSerializer是最基礎的序列化策略,其中「JacksonJsonRedisSerializer」與「OxmSerializer」都是基於stirng存儲,所以它們是較爲「高級」的序列化(最終仍是使用string解析以及構建java對象)。

    RedisTemplate中須要聲明4種serializer,默認爲「JdkSerializationRedisSerializer」:

    1) keySerializer :對於普通K-V操做時,key採起的序列化策略
    2) valueSerializer:value採起的序列化策略
    3) hashKeySerializer: 在hash數據結構中,hash-key的序列化策略
    4) hashValueSerializer:hash-value的序列化策略

    不管如何,建議key/hashKey採用StringRedisSerializer。

    接下來,經過實例描述如何使用它們,能夠首先參考「spring-data-redis特性」:

http://shift-alt-ctrl.iteye.com/blog/1887370
http://www.cnblogs.com/google4y/p/3535106.html
相關文章
相關標籤/搜索