在路上

 找回密码
 立即注册
在路上 站点首页 学习 查看内容

详解java之redis篇(spring-data-redis整合)

2017-3-7 12:51| 发布者: zhangjf| 查看: 1247| 评论: 0

摘要: 1,利用spring-data-redis整合 项目使用的pom.xml: project xmlns=http://maven.apache.org/POM/4.0.0 xmlns:xsi=http://www.w3.org/2001/XMLSchema-instance xsi:schemaLocation=http://maven.apache.org/POM/4.0 ...

1,利用spring-data-redis整合

项目使用的pom.xml:

  1. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  2. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  3. <modelVersion>4.0.0</modelVersion>
  4. <groupId>com.x.redis</groupId>
  5. <artifactId>Spring_redis</artifactId>
  6. <version>1.0-SNAPSHOT</version>
  7. <packaging>jar</packaging>
  8. <name>Spring_redis</name>
  9. <url>http://maven.apache.org</url>
  10. <properties>
  11. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  12. </properties>
  13. <dependencies>
  14. <dependency>
  15. <groupId>org.springframework.data</groupId>
  16. <artifactId>spring-data-redis</artifactId>
  17. <version>1.0.2.RELEASE</version>
  18. </dependency>
  19. <dependency>
  20. <groupId>org.springframework</groupId>
  21. <artifactId>spring-core</artifactId>
  22. <version>3.1.2.RELEASE</version>
  23. </dependency>
  24. <dependency>
  25. <groupId>redis.clients</groupId>
  26. <artifactId>jedis</artifactId>
  27. <version>2.1.0</version>
  28. </dependency>
  29. <dependency>
  30. <groupId>junit</groupId>
  31. <artifactId>junit</artifactId>
  32. <version>4.8.2</version>
  33. <scope>test</scope>
  34. </dependency>
  35. <dependency>
  36. <groupId>org.slf4j</groupId>
  37. <artifactId>slf4j-api</artifactId>
  38. <version>1.6.1</version>
  39. </dependency>
  40. <!-- 将现有的jakarta commons logging的调用转换成lsf4j的调用。 -->
  41. <dependency>
  42. <groupId>org.slf4j</groupId>
  43. <artifactId>jcl-over-slf4j</artifactId>
  44. <version>1.6.1</version>
  45. </dependency>
  46. <!-- Hack:确保commons-logging的jar包不被引入,否则将和jcl-over-slf4j冲突 -->
  47. <dependency>
  48. <groupId>commons-logging</groupId>
  49. <artifactId>commons-logging</artifactId>
  50. <version>1.1.1</version>
  51. <scope>provided</scope>
  52. </dependency>
  53. <!-- slf4j的实现:logback,用来取代log4j。更快、更强! -->
  54. <dependency>
  55. <groupId>ch.qos.logback</groupId>
  56. <artifactId>logback-classic</artifactId>
  57. <version>0.9.24</version>
  58. <scope>runtime</scope>
  59. </dependency>
  60. </dependencies>
  61. </project>
复制代码

除了log部分,只有一个spring core 和 spring-data-redis了

项目文件目录结构:

applicationContext.xml:

1,context:property-placeholder 标签用来导入properties文件。从而替换${redis.maxIdle}这样的变量。

2,context:component-scan 是为了在com.x.redis.dao报下的类能够实用spring的注解注入的方式。

3,事实上我们只需要把JedisPoolConfig配数来就好了,接下来就是spring的封装了。所以直接看UserDAOImpl的实现就明白了。

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx"
  6. xmlns:aop="http://www.springframework.org/schema/aop"
  7. xsi:schemaLocation="
  8. http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  9. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
  10. <context:property-placeholder location="classpath:redis.properties" />
  11. <context:component-scan base-package="com.x.redis.dao">
  12. </context:component-scan>
  13. <bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig">
  14. <property name="maxIdle" value="${redis.maxIdle}" />
  15. <property name="maxActive" value="${redis.maxActive}" />
  16. <property name="maxWait" value="${redis.maxWait}" />
  17. <property name="testOnBorrow" value="${redis.testOnBorrow}" />
  18. </bean>
  19. <bean id="connectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"
  20. p:host-name="${redis.host}" p:port="${redis.port}" p:password="${redis.pass}" p:pool-config-ref="poolConfig"/>
  21. <bean id="redisTemplate" class="org.springframework.data.redis.core.StringRedisTemplate">
  22. <property name="connectionFactory" ref="connectionFactory" />
  23. </bean>
  24. <bean id="userDAO" class="com.x.redis.dao.impl.UserDAOImpl" />
  25. </beans>
复制代码

redis.properties:

  1. # Redis settings
  2. #redis.host=192.168.20.101
  3. #redis.port=6380
  4. #redis.pass=foobared
  5. redis.host=127.0.0.1
  6. redis.port=6379
  7. redis.pass=
  8. redis.maxIdle=300
  9. redis.maxActive=600
  10. redis.maxWait=1000
  11. redis.testOnBorrow=true
复制代码

UserDAOImpl:

1,spring对dao层的封装很多用了类似于下面代码的模板方式。

2,RedisTemplate就是spring对redis的一个封装而已。

  1. public class UserDAOImpl implements UserDAO {
  2. @Autowired
  3. protected RedisTemplate<Serializable, Serializable> redisTemplate;
  4. public void saveUser(final User user) {
  5. redisTemplate.execute(new RedisCallback<Object>() {
  6. @Override
  7. public Object doInRedis(RedisConnection connection) throws DataAccessException {
  8. connection.set(redisTemplate.getStringSerializer().serialize("user.uid." + user.getId()),
  9. redisTemplate.getStringSerializer().serialize(user.getName()));
  10. return null;
  11. }
  12. });
  13. }
  14. @Override
  15. public User getUser(final long id) {
  16. return redisTemplate.execute(new RedisCallback<User>() {
  17. @Override
  18. public User doInRedis(RedisConnection connection) throws DataAccessException {
  19. byte[] key = redisTemplate.getStringSerializer().serialize("user.uid." + id);
  20. if (connection.exists(key)) {
  21. byte[] value = connection.get(key);
  22. String name = redisTemplate.getStringSerializer().deserialize(value);
  23. User user = new User();
  24. user.setName(name);
  25. user.setId(id);
  26. return user;
  27. }
  28. return null;
  29. }
  30. });
  31. }
  32. }
复制代码

其他:

User:

  1. public class User {
  2. private long id;
  3. private String name;
  4. public long getId() {
  5. return id;
  6. }
  7. public void setId(long id) {
  8. this.id = id;
  9. }
  10. public String getName() {
  11. return name;
  12. }
  13. public void setName(String name) {
  14. this.name = name;
  15. }
  16. }
复制代码

测试代码:

  1. public static void main(String[] args) {
  2. ApplicationContext ac = new ClassPathXmlApplicationContext("classpath:/applicationContext.xml");
  3. UserDAO userDAO = (UserDAO)ac.getBean("userDAO");
  4. User user1 = new User();
  5. user1.setId(1);
  6. user1.setName("obama");
  7. userDAO.saveUser(user1);
  8. User user2 = userDAO.getUser(1);
  9. System.out.println(user2.getName());
  10. }
复制代码

2,不利用spring-data-redis整合

个人觉得这样整合灵活度更大,能够更加明了的完成任务。

pom.xml:

  1. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  2. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  3. <modelVersion>4.0.0</modelVersion>
  4. <groupId>com.d.work</groupId>
  5. <artifactId>Redis_Templete</artifactId>
  6. <version>1.0-SNAPSHOT</version>
  7. <packaging>jar</packaging>
  8. <name>Redis_Templete</name>
  9. <url>http://maven.apache.org</url>
  10. <properties>
  11. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  12. </properties>
  13. <dependencies>
  14. <dependency>
  15. <groupId>junit</groupId>
  16. <artifactId>junit</artifactId>
  17. <version>3.8.1</version>
  18. <scope>test</scope>
  19. </dependency>
  20. <dependency>
  21. <groupId>redis.clients</groupId>
  22. <artifactId>jedis</artifactId>
  23. <version>2.1.0</version>
  24. </dependency>
  25. <dependency>
  26. <groupId>org.springframework</groupId>
  27. <artifactId>spring-core</artifactId>
  28. <version>3.1.2.RELEASE</version>
  29. </dependency>
  30. <dependency>
  31. <groupId>org.springframework</groupId>
  32. <artifactId>spring-beans</artifactId>
  33. <version>3.1.2.RELEASE</version>
  34. </dependency>
  35. <dependency>
  36. <groupId>org.springframework</groupId>
  37. <artifactId>spring-context</artifactId>
  38. <version>3.1.2.RELEASE</version>
  39. </dependency>
  40. <dependency>
  41. <groupId>org.slf4j</groupId>
  42. <artifactId>slf4j-api</artifactId>
  43. <version>1.6.1</version>
  44. </dependency>
  45. <!-- 将现有的jakarta commons logging的调用转换成lsf4j的调用。 -->
  46. <dependency>
  47. <groupId>org.slf4j</groupId>
  48. <artifactId>jcl-over-slf4j</artifactId>
  49. <version>1.6.1</version>
  50. </dependency>
  51. <!-- Hack:确保commons-logging的jar包不被引入,否则将和jcl-over-slf4j冲突 -->
  52. <dependency>
  53. <groupId>commons-logging</groupId>
  54. <artifactId>commons-logging</artifactId>
  55. <version>1.1.1</version>
  56. <scope>provided</scope>
  57. </dependency>
  58. <!-- slf4j的实现:logback,用来取代log4j。更快、更强! -->
  59. <dependency>
  60. <groupId>ch.qos.logback</groupId>
  61. <artifactId>logback-classic</artifactId>
  62. <version>0.9.24</version>
  63. <scope>runtime</scope>
  64. </dependency>
  65. </dependencies>
  66. </project>
复制代码

目录结构:

data-source.xml

1,context:property-placeholder 和 context:component-scan 前面解释过啦。

2,配置了一个ShardedJedisPool,在jdeis里 还有个JedisPool。这两个的区别:

一个是分片形式,可以连接有主备的redis服务端,一个是单个的。详细后续学习

3,因为不使用spring-data-redis的封装,所以自己要自己封装一个

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx"
  6. xmlns:aop="http://www.springframework.org/schema/aop"
  7. xsi:schemaLocation="
  8. http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  9. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
  10. <context:property-placeholder location="classpath:redis.properties" />
  11. <context:component-scan base-package="com.d.work.main">
  12. </context:component-scan>
  13. <context:component-scan base-package="com.d.work.redis">
  14. </context:component-scan>
  15. <bean id="jedisPoolConfig" class="redis.clients.jedis.JedisPoolConfig">
  16. <property name="maxActive" value="50" />
  17. <property name="maxIdle" value="8" />
  18. <property name="maxWait" value="1000" />
  19. <property name="testOnBorrow" value="true"/>
  20. <property name="testOnReturn" value="true"/>
  21. <!-- <property name="testWhileIdle" value="true"/> -->
  22. </bean>
  23. <bean id="shardedJedisPool" class="redis.clients.jedis.ShardedJedisPool" scope="singleton">
  24. <constructor-arg index="0" ref="jedisPoolConfig" />
  25. <constructor-arg index="1">
  26. <list>
  27. <bean class="redis.clients.jedis.JedisShardInfo">
  28. <constructor-arg name="host" value="${redis.host}" />
  29. <constructor-arg name="port" value="${redis.port}" />
  30. <constructor-arg name="timeout" value="${redis.timeout}" />
  31. <constructor-arg name="weight" value="1" />
  32. </bean>
  33. </list>
  34. </constructor-arg>
  35. </bean>
  36. </beans>
复制代码

RedisDataSource:定义三个方法

  1. public interface RedisDataSource {
  2. public abstract ShardedJedis getRedisClient();
  3. public void returnResource(ShardedJedis shardedJedis);
  4. public void returnResource(ShardedJedis shardedJedis,boolean broken);
  5. }
复制代码

实现redisDataSource:

1, 注入配置好的ShardedJedisPool,这三个方法的作用:

  • getRedisClient() : 取得redis的客户端,可以执行命令了。
  • returnResource(ShardedJedis shardedJedis) : 将资源返还给pool
  • returnResource(ShardedJedis shardedJedis, boolean broken) : 出现异常后,将资源返还给pool (其实不需要第二个方法)
  1. @Repository("redisDataSource")
  2. public class RedisDataSourceImpl implements RedisDataSource {
  3. private static final Logger log = LoggerFactory.getLogger(RedisDataSourceImpl.class);
  4. @Autowired
  5. private ShardedJedisPool shardedJedisPool;
  6. public ShardedJedis getRedisClient() {
  7. try {
  8. ShardedJedis shardJedis = shardedJedisPool.getResource();
  9. return shardJedis;
  10. } catch (Exception e) {
  11. log.error("getRedisClent error", e);
  12. }
  13. return null;
  14. }
  15. public void returnResource(ShardedJedis shardedJedis) {
  16. shardedJedisPool.returnResource(shardedJedis);
  17. }
  18. public void returnResource(ShardedJedis shardedJedis, boolean broken) {
  19. if (broken) {
  20. shardedJedisPool.returnBrokenResource(shardedJedis);
  21. } else {
  22. shardedJedisPool.returnResource(shardedJedis);
  23. }
  24. }
  25. }
复制代码

第二层的封装:RedisClientTemplate,例子实现了放值和取值。最后代码提供了全部命令的实现。

代码就是映射性质的又一次调用jedis的方法而已,用了个broken来做标示符,决定返还资源的方式。

这一层的目的主要也是让再上层的调用不需要关心pool中链接的取得和返还问题了。

  1. @Repository("redisClientTemplate")
  2. public class RedisClientTemplate {
  3. private static final Logger log = LoggerFactory.getLogger(RedisClientTemplate.class);
  4. @Autowired
  5. private RedisDataSource redisDataSource;
  6. public void disconnect() {
  7. ShardedJedis shardedJedis = redisDataSource.getRedisClient();
  8. shardedJedis.disconnect();
  9. }
  10. /**
  11. * 设置单个值
  12. *
  13. * @param key
  14. * @param value
  15. * @return
  16. */
  17. public String set(String key, String value) {
  18. String result = null;
  19. ShardedJedis shardedJedis = redisDataSource.getRedisClient();
  20. if (shardedJedis == null) {
  21. return result;
  22. }
  23. boolean broken = false;
  24. try {
  25. result = shardedJedis.set(key, value);
  26. } catch (Exception e) {
  27. log.error(e.getMessage(), e);
  28. broken = true;
  29. } finally {
  30. redisDataSource.returnResource(shardedJedis, broken);
  31. }
  32. return result;
  33. }
  34. /**
  35. * 获取单个值
  36. *
  37. * @param key
  38. * @return
  39. */
  40. public String get(String key) {
  41. String result = null;
  42. ShardedJedis shardedJedis = redisDataSource.getRedisClient();
  43. if (shardedJedis == null) {
  44. return result;
  45. }
  46. boolean broken = false;
  47. try {
  48. result = shardedJedis.get(key);
  49. } catch (Exception e) {
  50. log.error(e.getMessage(), e);
  51. broken = true;
  52. } finally {
  53. redisDataSource.returnResource(shardedJedis, broken);
  54. }
  55. return result;
  56. }
  57. }
复制代码

测试代码:

  1. public static void main(String[] args) {
  2. ApplicationContext ac = new ClassPathXmlApplicationContext("classpath:/data-source.xml");
  3. RedisClientTemplate redisClient = (RedisClientTemplate)ac.getBean("redisClientTemplate");
  4. redisClient.set("a", "abc");
  5. System.out.println(redisClient.get("a"));
  6. }
复制代码

附上RedisClientTemplate全部实现:

RedisClientTemplate代码太多,附上下载地址:http://xiazai.jb51.net/201701/yuanma/RedisClientTemplate_jb51.rar

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持程序员之家。

最新评论

小黑屋|在路上 ( 蜀ICP备15035742号-1 

;

GMT+8, 2025-5-4 02:20

Copyright 2015-2025 djqfx

返回顶部