年末将至,是时候该把所学的总结下了。最近正好从eclipes转到idea,发现idea对模组的支持很棒。这一片先总结下springboot和redis的整合
首先添加redis服务器
直接用docker 远程拉取即可,这里不再描述
直接撸代码
1.添加依赖
1 <dependency>
2 <groupId>org.springframework.boot</groupId>
3 <artifactId>spring-boot-starter-data-redis</artifactId>
4 </dependency>
2.在配置文件中增加配置的连接信息
## Redis 配置 ## Redis数据库索引(默认为0) spring.redis.database=0 ## Redis服务器地址 spring.redis.host=127.0.0.1 ## Redis服务器连接端口 spring.redis.port=6379 ## Redis服务器连接密码(默认为空) spring.redis.password= ## 连接池最大连接数(使用负值表示没有限制) spring.redis.pool.max-active=8 ## 连接池最大阻塞等待时间(使用负值表示没有限制) spring.redis.pool.max-wait=-1 ## 连接池中的最大空闲连接 spring.redis.pool.max-idle=8 ## 连接池中的最小空闲连接 spring.redis.pool.min-idle=0 ## 连接超时时间(毫秒) spring.redis.timeout=0
好了,基本的配置就好了
我们可以写个测试类,看看调用情况
package com.hdkj.redis;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest
public class RedisApplicationTests {
@Autowired
private StringRedisTemplate stringRedisTemplate;
@Test
public void contextLoads() {
stringRedisTemplate.opsForValue().set("xmz","23");
String xmz = stringRedisTemplate.opsForValue().get("xmz");
System.out.println(xmz);
}
}
输出结果:23
好了,这就是最简单的配置了。
======================================================================================================
springboot更推荐我们把配置信息写在类里面
我们可以对以上的配置信息改造下
新建一个配置实体类 RedisProperties
1 package com.hdkj.redis.config;
2
3 import org.springframework.boot.context.properties.ConfigurationProperties;
4 import org.springframework.context.annotation.PropertySource;
5
6 import java.util.List;
7
8 /**
9 * @author xuminzhe
10 * @version V1.0
11 * @Project study
12 * @Package com.hdkj.redis.config
13 * @Description
14 * @Date 2017/12/4
15 */
16 @ConfigurationProperties(prefix = "spring.redis")
17 //@PropertySource("classpath:application.properties")
18 public class RedisProperties {
19 /**
20 * Database index used by the connection factory.
21 */
22 private int database = 0;
23
24 /**
25 * Redis url, which will overrule host, port and password if set.
26 */
27 private String url;
28
29 /**
30 * Redis server host.
31 */
32 private String host = "localhost";
33
34 /**
35 * Login password of the redis server.
36 */
37 private String password;
38
39 /**
40 * Redis server port.
41 */
42 private int port = 6379;
43
44 /**
45 * Enable SSL.
46 */
47 private boolean ssl;
48
49 /**
50 * Connection timeout in milliseconds.
51 */
52 private int timeout;
53
54 private RedisProperties.Pool pool;
55
56 private RedisProperties.Sentinel sentinel;
57
58 private RedisProperties.Cluster cluster;
59
60 public int getDatabase() {
61 return this.database;
62 }
63
64 public void setDatabase(int database) {
65 this.database = database;
66 }
67
68 public String getUrl() {
69 return this.url;
70 }
71
72 public void setUrl(String url) {
73 this.url = url;
74 }
75
76 public String getHost() {
77 return this.host;
78 }
79
80 public void setHost(String host) {
81 this.host = host;
82 }
83
84 public String getPassword() {
85 return this.password;
86 }
87
88 public void setPassword(String password) {
89 this.password = password;
90 }
91
92 public int getPort() {
93 return this.port;
94 }
95
96 public void setPort(int port) {
97 this.port = port;
98 }
99
100 public boolean isSsl() {
101 return this.ssl;
102 }
103
104 public void setSsl(boolean ssl) {
105 this.ssl = ssl;
106 }
107
108 public void setTimeout(int timeout) {
109 this.timeout = timeout;
110 }
111
112 public int getTimeout() {
113 return this.timeout;
114 }
115
116 public RedisProperties.Sentinel getSentinel() {
117 return this.sentinel;
118 }
119
120 public void setSentinel(RedisProperties.Sentinel sentinel) {
121 this.sentinel = sentinel;
122 }
123
124 public RedisProperties.Pool getPool() {
125 return this.pool;
126 }
127
128 public void setPool(RedisProperties.Pool pool) {
129 this.pool = pool;
130 }
131
132 public RedisProperties.Cluster getCluster() {
133 return this.cluster;
134 }
135
136 public void setCluster(RedisProperties.Cluster cluster) {
137 this.cluster = cluster;
138 }
139
140 /**
141 * Pool properties.
142 */
143 public static class Pool {
144
145 /**
146 * Max number of "idle" connections in the pool. Use a negative value to indicate
147 * an unlimited number of idle connections.
148 */
149 private int maxIdle = 8;
150
151 /**
152 * Target for the minimum number of idle connections to maintain in the pool. This
153 * setting only has an effect if it is positive.
154 */
155 private int minIdle = 0;
156
157 /**
158 * Max number of connections that can be allocated by the pool at a given time.
159 * Use a negative value for no limit.
160 */
161 private int maxActive = 8;
162
163 /**
164 * Maximum amount of time (in milliseconds) a connection allocation should block
165 * before throwing an exception when the pool is exhausted. Use a negative value
166 * to block indefinitely.
167 */
168 private int maxWait = -1;
169
170 public int getMaxIdle() {
171 return this.maxIdle;
172 }
173
174 public void setMaxIdle(int maxIdle) {
175 this.maxIdle = maxIdle;
176 }
177
178 public int getMinIdle() {
179 return this.minIdle;
180 }
181
182 public void setMinIdle(int minIdle) {
183 this.minIdle = minIdle;
184 }
185
186 public int getMaxActive() {
187 return this.maxActive;
188 }
189
190 public void setMaxActive(int maxActive) {
191 this.maxActive = maxActive;
192 }
193
194 public int getMaxWait() {
195 return this.maxWait;
196 }
197
198 public void setMaxWait(int maxWait) {
199 this.maxWait = maxWait;
200 }
201
202 }
203
204 /**
205 * Cluster properties.
206 */
207 public static class Cluster {
208
209 /**
210 * Comma-separated list of "host:port" pairs to bootstrap from. This represents an
211 * "initial" list of cluster nodes and is required to have at least one entry.
212 */
213 private List<String> nodes;
214
215 /**
216 * Maximum number of redirects to follow when executing commands across the
217 * cluster.
218 */
219 private Integer maxRedirects;
220
221 public List<String> getNodes() {
222 return this.nodes;
223 }
224
225 public void setNodes(List<String> nodes) {
226 this.nodes = nodes;
227 }
228
229 public Integer getMaxRedirects() {
230 return this.maxRedirects;
231 }
232
233 public void setMaxRedirects(Integer maxRedirects) {
234 this.maxRedirects = maxRedirects;
235 }
236
237 }
238
239 /**
240 * Redis sentinel properties.
241 */
242 public static class Sentinel {
243
244 /**
245 * Name of Redis server.
246 */
247 private String master;
248
249 /**
250 * Comma-separated list of host:port pairs.
251 */
252 private String nodes;
253
254 public String getMaster() {
255 return this.master;
256 }
257
258 public void setMaster(String master) {
259 this.master = master;
260 }
261
262 public String getNodes() {
263 return this.nodes;
264 }
265
266 public void setNodes(String nodes) {
267 this.nodes = nodes;
268 }
269
270 }
271
272 }
如果熟悉mybatis配置的朋友一定对这种方式不会陌生,接下来我们在新建一个配置类
1 package com.hdkj.redis.config;
2
3 import org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration;
4 import org.springframework.boot.context.properties.EnableConfigurationProperties;
5 import org.springframework.cache.annotation.CachingConfigurerSupport;
6 import org.springframework.cache.annotation.EnableCaching;
7 import org.springframework.context.annotation.Bean;
8 import org.springframework.context.annotation.Configuration;
9 import org.springframework.context.annotation.Import;
10 import org.springframework.data.redis.connection.RedisConnectionFactory;
11 import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
12 import org.springframework.data.redis.core.RedisTemplate;
13 import redis.clients.jedis.JedisPoolConfig;
14 import redis.clients.jedis.JedisShardInfo;
15 import redis.clients.jedis.ShardedJedisPool;
16
17 import javax.inject.Inject;
18 import java.util.ArrayList;
19 import java.util.List;
20
21 /**
22 * @author xuminzhe
23 * @version V1.0
24 * @Project study
25 * @Package com.hdkj.redis.config
26 * @Description
27 * @Date 2017/12/4
28 */
29 @Configuration
30 @EnableCaching
31 @Import(value = RedisAutoConfiguration.class)
32 @EnableConfigurationProperties(RedisProperties.class)
33 public class RedisConfiguration extends CachingConfigurerSupport {
34 @Inject
35 private RedisProperties properties;
36
37 @Bean
38 public RedisConnectionFactory jedisConnectionFactory() {
39 JedisPoolConfig poolConfig = new JedisPoolConfig();
40 poolConfig.setMaxTotal(properties.getPool().getMaxActive());
41 poolConfig.setMaxIdle(properties.getPool().getMaxIdle());
42 poolConfig.setMaxWaitMillis(properties.getPool().getMaxWait());
43 poolConfig.setTestOnBorrow(true);
44 poolConfig.setTestOnCreate(true);
45 poolConfig.setTestWhileIdle(true);
46 JedisConnectionFactory jedisConnectionFactory = new JedisConnectionFactory(poolConfig);
47 jedisConnectionFactory.setHostName(properties.getHost());
48 if (null != properties.getPassword()) {
49 jedisConnectionFactory.setPassword(properties.getPassword());
50 }
51 jedisConnectionFactory.setDatabase(properties.getDatabase());
52 jedisConnectionFactory.setPort(properties.getPort());
53
54 //其他配置,可再次扩展
55 return jedisConnectionFactory;
56 }
57
58 @Bean(name = "redisTemplate")
59 public RedisTemplate redisTemplate() {
60 RedisTemplate redisTemplate = new RedisTemplate();
61
62 redisTemplate.setConnectionFactory(jedisConnectionFactory());
63
64 // redisTemplate.setHashValueSerializer(new StringRedisSerializer());
65 // redisTemplate.setKeySerializer(new StringRedisSerializer());
66 // redisTemplate.setValueSerializer(new StringRedisSerializer());
67 // redisTemplate.setHashKeySerializer(new StringRedisSerializer());
68
69 redisTemplate.afterPropertiesSet();
70 redisTemplate.setEnableTransactionSupport(true);
71
72 return redisTemplate;
73 }
74
75 /**
76 * 分布式环境下多台redis的配置
77 * @return
78 */
79 @Bean
80 public ShardedJedisPool shardedJedisPool(){
81 JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
82 jedisPoolConfig.setMaxTotal(properties.getPool().getMaxActive());
83 jedisPoolConfig.setMaxIdle(properties.getPool().getMaxIdle());
84 jedisPoolConfig.setMaxWaitMillis(properties.getPool().getMaxWait());
85 List<JedisShardInfo> jedisShardInfoList = new ArrayList<>();
86 JedisShardInfo jedisShardInfo = new JedisShardInfo(properties.getHost(), properties.getPort());
87 jedisShardInfo.setPassword(properties.getPassword());
88 jedisShardInfoList.add(jedisShardInfo);
89 return new ShardedJedisPool(jedisPoolConfig, jedisShardInfoList);
90 }
91
92
93 }
基本的配置就完成了,这里我们再配置一个工具类用于常规的redis map的操作
package com.hdkj.redis.utils;
import org.springframework.data.redis.core.HashOperations;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
/**
* @author xuminzhe
* @version V1.0
* @Project study
* @Package com.hdkj.redis.utils
* @Description redis工具类
* @Date 2017/12/4
*/
@Component("mapRedisTemplate")
public class MapRedisTemplate {
@Resource(name = "redisTemplate")
private RedisTemplate<String,Object> redisTemplate;
public Boolean hasKey(String key, Object hashKey) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
return ops.hasKey(key, hashKey);
}
public Boolean delete(String key) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
Long l = ops.delete(key);
return l == 1 ? Boolean.TRUE : Boolean.FALSE;
}
public Boolean delete(String key, Object hashKey) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
Long l = ops.delete(key, hashKey);
return l > 0 ? Boolean.TRUE : Boolean.FALSE;
}
public Object get(String key, Object hashKey) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
return ops.get(key, hashKey);
}
public Map<Object, Object> entries(String key) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
return ops.entries(key);
}
public Set<Object> keys(String key) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
return ops.keys(key);
}
public List<Object> values(String key) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
return ops.values(key);
}
public Long size(String key) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
return ops.size(key);
}
public void put(String key, Object hashKey, Object value) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
ops.put(key, hashKey, value);
}
public void put(String key, Object hashKey, Object value,Long expireIn) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
ops.put(key, hashKey, value);
redisTemplate.expire(key, expireIn, TimeUnit.MILLISECONDS);
}
public void putIfAbsent(String key, Object hashKey, Object value) {
HashOperations<String, Object, Object> ops = redisTemplate.opsForHash();
ops.putIfAbsent(key, hashKey, value);
}
}
好了,这就是一个简单的redis项目的配置。
======================================================================================================================================
有个问题还没解决:其他模块在调用redis模块,读取的是自己本身模块路劲下的配置文件信息,所以在调用的时候老是会失败,我暂时的处理方式是把配置信息写到 其他模块中。但是这种方式太不优雅了,有哪位朋友遇到相同情况的可以交流下。
======================================================================================================================================
项目在做自增的时候报 nested exception is redis.clients.jedis.exceptions.JedisDataException: ERR value is not an integer or out of range
发现redis 默认的序列化是JdkSerializationRedisSerializer,其将初始值变成了序列化字符串存入了Redis,而Redis执行INCRBY命令时是无法识别序列化字符串为整数的。从而导致以上错误
所以我们要修改为项目中序列化的方式