1. 程式人生 > >Spring整合Redis實現資料快取

Spring整合Redis實現資料快取

一、什麼是Redis

Redis 是一個開源(BSD許可)的,記憶體中的資料結構儲存系統,它可以用作資料庫、快取和訊息中介軟體。 它支援多種型別的資料結構,如 字串(strings), 雜湊(hashes), 列表(lists), 集合(sets), 有序集合(sorted sets) 與範圍查詢, bitmaps, hyperloglogs 和 地理空間(geospatial) 索引半徑查詢。 Redis 內建了 複製(replication),LUA指令碼(Lua scripting), LRU驅動事件(LRU eviction),事務(transactions) 和不同級別的 磁碟持久化(persistence), 並通過 Redis哨兵(Sentinel)和自動 分割槽(Cluster)提供高可用性(high availability)。

詳細請看:http://www.redis.cn/

二、RedisTemplate

在Spring中,我們對Redis儲存的資料進行操作時,通常使用的是RedisTemplate這個類。

我們知道,在對Redis資料進行訪問時,一般情況下都是使用Redis命令進行訪問,但是在Spring中,為了簡化對Redis資料的訪問,就產生了RedisTemplate。

RedisTemplate是一個簡化Redis資料訪問的一個幫助類,此類對Redis命令進行高階封裝,通過此類可以呼叫ValueOperations和ListOperations等等方法。

三、安裝Redis

Redis 使用 ANSI C 編寫並且能在絕大Linux系統上執行,基於BSD協議,對OS X沒有外部依賴.

支援Linux 和 OS X兩種系統的開發和測試,並且也推薦使用Linux部署. Redis 可以像SmartOS一樣執行在Solaris系統中, 但是需要注意的是官方不支援Windos版本的Redis,但微軟開發和維護著支援win-64 的Redis版本.

windows版本Redis下載連結:https://github.com/MSOpenTech/redis

下載之後按照一般安裝方式預設安裝就可以了,安裝完之後啟動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.0 http://maven.apache.org/maven-v4_0_0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<groupId>com.redis</groupId>
	<artifactId>rediscache</artifactId>
	<packaging>war</packaging>
	<version>0.0.1-SNAPSHOT</version>
	<name>rediscache Maven Webapp</name>
	<url>http://maven.apache.org</url>

	<properties>
		<!-- spring版本號 -->
		<spring.version>4.3.2.RELEASE</spring.version>
	</properties>

	<dependencies>
		<dependency>
			<groupId>junit</groupId>
			<artifactId>junit</artifactId>
			<version>3.8.1</version>
			<scope>test</scope>
		</dependency>
		  <!-- 新增Spring依賴 -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context-support</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aspects</artifactId>
            <version>${spring.version}</version>
        </dependency>
		<!--spring單元測試依賴 -->
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-test</artifactId>
			<version>${spring.version}</version>
			<scope>test</scope>
		</dependency>
		<!-- Redis 相關依賴 -->
		<dependency>
			<groupId>org.springframework.data</groupId>
			<artifactId>spring-data-redis</artifactId>
			<version>1.6.1.RELEASE</version>
		</dependency>
		<dependency>
			<groupId>redis.clients</groupId>
			<artifactId>jedis</artifactId>
			<version>2.7.3</version>
		</dependency>
	</dependencies>
	<build>
		<finalName>rediscache</finalName>
	</build>
</project>
這裡我們依賴了spring-aop和spring-aspect,主要是我們在單元測試的時候需要使用,如果不使用單元測試可以不依賴spring-aop和spring-aspect.

五、在resources資料夾下新建config.properties檔案並寫入Redis相關配置資訊

#redis中心
#繫結的主機地址
redis.host=127.0.0.1
#指定Redis監聽埠,預設埠為6379
redis.port=6379 
#授權密碼(本例子沒有使用)
redis.password=123456  
#最大空閒數:空閒連結數大於maxIdle時,將進行回收
redis.maxIdle=100  
#最大連線數:能夠同時建立的“最大連結個數”
redis.maxActive=300  
#最大等待時間:單位ms
redis.maxWait=1000   
#使用連線時,檢測連線是否成功 
redis.testOnBorrow=true 
#當客戶端閒置多長時間後關閉連線,如果指定為0,表示關閉該功能
redis.timeout=10000
六、在resources資料夾下新建spring-mvc.xml並配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
	xmlns:context="http://www.springframework.org/schema/context"
	xmlns:mvc="http://www.springframework.org/schema/mvc"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
       http://www.springframework.org/schema/context 
       http://www.springframework.org/schema/context/spring-context-4.0.xsd
       http://www.springframework.org/schema/mvc
       http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd">

	<!-- 自動掃描註解的bean -->
	<context:component-scan base-package="rediscache.service" />
	
	<!-- 引入properties配置檔案 --> 
	<context:property-placeholder ignore-unresolvable="true" location="classpath:config.properties" />
    
    <!-- jedis 配置 -->
    <bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig" >
          <property name="maxIdle" value="${redis.maxIdle}" />
          <property name="maxWaitMillis" value="${redis.maxWait}" />
          <property name="testOnBorrow" value="${redis.testOnBorrow}" />
    </bean>

    <!-- redis伺服器中心 -->
    <bean id="connectionFactory"  class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory" >
          <property name="poolConfig" ref="poolConfig" />
          <property name="port" value="${redis.port}" />
          <property name="hostName" value="${redis.host}" />
          <!-- <property name="password" value="${redis.password}" /> -->
          <property name="timeout" value="${redis.timeout}" ></property>
    </bean >
    
    <bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate" >
          <property name="connectionFactory" ref="connectionFactory" />
          <property name="keySerializer" >
              <bean class="org.springframework.data.redis.serializer.StringRedisSerializer" />
          </property>
          <property name="valueSerializer" >
              <bean class="org.springframework.data.redis.serializer.JdkSerializationRedisSerializer" />
          </property>
    </bean>
    <bean id="redisUtil" class="rediscache.utils.RedisUtil" />
</beans>

七、新建rediscache.utils包並實現RedisUtil類
package rediscache.utils;

import java.io.Serializable;
import java.util.Set;
import java.util.concurrent.TimeUnit;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.HashOperations;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.ValueOperations;

/**
 * @Comment
 * @Author Ron
 * @date 2017年1月13日 下午5:24:33
 */
public class RedisUtil {
	/**
	 * RedisTemplate是一個簡化Redis資料訪問的一個幫助類,
	 * 此類對Redis命令進行高階封裝,通過此類可以呼叫ValueOperations和ListOperations等等方法。
	 */
	@Autowired
	private RedisTemplate<Serializable, Object> redisTemplate;

	/**
	 * 批量刪除對應的value
	 * 
	 * @param keys
	 */
	public void remove(final String... keys) {
		for (String key : keys) {
			remove(key);
		}
	}

	/**
	 * 批量刪除key
	 * 
	 * @param pattern
	 */
	public void removePattern(final String pattern) {
		Set<Serializable> keys = redisTemplate.keys(pattern);
		if (keys.size() > 0)
			redisTemplate.delete(keys);
	}

	/**
	 * 刪除對應的value
	 * @param key
	 */
	public void remove(final String key) {
		if (exists(key)) {
			redisTemplate.delete(key);
		}
	}

	/**
	 *
	 * @param key
	 * @return
	 */
	public boolean exists(final String key) {
		return redisTemplate.hasKey(key);
	}

	/**
	 * 讀取快取
	 * @param key
	 * @return
	 */
	public Object get(final String key) {
		Object result = null;
		ValueOperations<Serializable, Object> operations = redisTemplate.opsForValue();
		result = operations.get(key);
		return result;
	}
	
	/**
	 * 
	 * @Author zg
	 * @Date 2016年12月15日 上午11:28:46
	 * @param key
	 * @param hashKey
	 * @return
	 */
	public Object get(final String key, final String hashKey){
		Object result = null;
		HashOperations<Serializable,Object,Object> operations = redisTemplate.opsForHash();
		result = operations.get(key, hashKey);
		return result;
	}

	/**
	 * 寫入快取
	 * 
	 * @param key
	 * @param value
	 * @return
	 */
	public boolean set(final String key, Object value) {
		boolean result = false;
		try {
			ValueOperations<Serializable, Object> operations = redisTemplate.opsForValue();
			operations.set(key, value);
			result = true;
		} catch (Exception e) {
			e.printStackTrace();
		}
		return result;
	}
	
	/**
	 * 
	 * @Author Ron
	 * @param key
	 * @param hashKey
	 * @param value
	 * @return
	 */
	public boolean set(final String key, final String hashKey, Object value) {
		boolean result = false;
		try {
			HashOperations<Serializable,Object,Object> operations = redisTemplate.opsForHash();
			operations.put(key, hashKey, value);
			result = true;
		} catch (Exception e) {
			e.printStackTrace();
		}
		return result;
	}

	/**
	 * 寫入快取
	 * 
	 * @param key
	 * @param value
	 * @return
	 */
	public boolean set(final String key, Object value, Long expireTime) {
		boolean result = false;
		try {
			ValueOperations<Serializable, Object> operations = redisTemplate.opsForValue();
			operations.set(key, value);
			redisTemplate.expire(key, expireTime, TimeUnit.SECONDS);
			result = true;
		} catch (Exception e) {
			e.printStackTrace();
		}
		return result;
	}
}
八、新建rediscache.service包並實現服務測試類RedisTestServiceImpl
package rediscache.service;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import rediscache.utils.RedisUtil;

/**
 * @Comment
 * @Author Ron
 * @date 2017年1月13日 下午4:52:55
 */
@Component("redisTestService")
public class RedisTestServiceImpl {
	
	@Autowired
	RedisUtil redisUtil;
	
	public boolean setValue(String key,String value) {
		try {
			redisUtil.set(key, value);
			return true;
		} catch (Exception e) {
			return false;
		}
	}
	
	public String getValue(String key) {
		try {
			String value = (String) redisUtil.get(key);
			return value;
		} catch (Exception e) {
			return "讀取快取出錯。。。";
		}
	}
}
九、編寫單元測試
package rediscache;

import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import rediscache.service.RedisTestServiceImpl;

/**
 * @Comment
 * @Author Ron
 * @date 2017年1月13日 下午4:43:15
 */
public class testCache {
	private ClassPathXmlApplicationContext context;
	
	@Autowired
	private RedisTestServiceImpl redisTestService;
	
	@Before
	public void initConfig(){
		context = new ClassPathXmlApplicationContext("classpath:spring-mvc.xml");
		redisTestService = (RedisTestServiceImpl) context.getBean("redisTestService");
	}

	@After
	public void end(){
		if(context != null){
			context.close();
		}
	}
	
	@Test
	public void testRedisCache(){
		redisTestService.setValue("redis", "I love you redis!");
		System.out.println(redisTestService.getValue("redis"));
		System.out.println(redisTestService.getValue("redis"));
		System.out.println(redisTestService.getValue("redis"));
		System.out.println(redisTestService.getValue("redis"));
	}
}


十,測試結果