1. 什么是Redis
Redis是使用C语言开发的支持网络、基于内存可持久化的日志型的一个开源的高性能键值对(key-value)数据库。目前为止,Redis支持5种键值数据类型,分别是字符串类型(String)、散列类型(Hash)、列表类型(List)、集合类型(Set)和有序集合类型(SortedSet)。
2. Redis的安装与连接
2.1. 下载与安装
在不同系统环境下,安装方式不同,这里只介绍Linux系统下的安装方式。需要注意的是,Redis的安装需要使用C语言的编译环境,所以需要提前安装编译环境,具体操作步骤如下:
第一步:使用yum进行C语言环境的在线安装
image.png
第二步:检查是否安装成功,输入gcc,出现gcc:no input files 表示安装成功。
image.png第三步:下载Redis压缩包
第四步:拷贝压缩包到Linux系统上,并解压。
第五步:进行编译安装,cd到Redis解压目录,输入命令进行安装。输入命令:make install PREFIX=/usr/local/redis,注意PREFIX 必须是大写的。当出现如下图所示,表示安装成功。
image.png2.2. Redis服务两种启动模式
2.2.1 前端启动
进入到redis的安装目录下的bin目录/usr/local/redis/bin,输入命令./redis-server,进行启动,出现下图所示,表示启动成功。
image.png2.2.2 后台启动
第一步:把redis解压包下面的redis.conf文件拷贝到/usr/local/redis/bin目录下。
image.png第二步:使用vim命令修改redis.conf配置文件,将daemonize no 修改为daemonize yes
image.png第三步:将protected-mode yes改为protected-mode no,否则外部访问会被拒绝。
image.png第四步:将bind 127.0.0.1语句注释掉(该语句前面加#注释),否则只能本地访问,外部ip不能访问。
image.png第五步:输入启动命令,并检查启动是否成功,在进程发现redis服务表示启动成功。
image.png2.3. 服务的连接与关闭
2.3.1 连接Redis服务
适用redis-cli建立连接,输入如下命令:
-h:表示连接的服务器地址
-p:表示连接的端口
image.png2.3.2 关闭Redis服务
关闭服务有两种方式,第一种是通过连接上的客户端进行关闭,使用shutdown命令进行关闭。
image.png第二种,是用kill命令杀死进程进行关闭,找到对应Redis服务的进程id,输入命令进行关闭。kill -9 pid(pid为进程id)
image.png3. Redis的两种持久化方案
3.1. RDB快照形式
在Redis服务中,默认开启的就是RDB快照模式,会定期将当前时刻的数据保存到dump.rdb的文件中,但是保存周期较长,容易存在数据丢失,对服务器性能要求较低。如下图所示,redis.conf中默认设置了RDB模式的保存规则和间隔时间。
image.png3.2. AOF形式
AOF,全称append only file,是指对所有redis的操作命令都会记录在appendonly.aof的文件中,对数据保存比较完整,每秒进行保存一次,但比较耗费性能。若要开启AOF模式,需对redis.conf文件进行修改,将appendonly设置为yes,如图所示:
image.png若同时开启两个持久化方案,则会按照AOF的持久化方案进行数据恢复。
4. Redis集群的搭建和配置
搭建Redis集群至少需要3个节点,为了提高集群的高可用性,还需要为每一个节点增加一台备份机,总共需要6台服务器。
第一步:在每台服务器上安装Redis,具体安装步骤参考第2章。
第二步:在每台服务器上安装集群需要的gem包,安装命名为:
image.png第三步:修改每台服务器上面的redis.conf配置文件,将cluster-enable yes前面的注释去掉。
image.png第四步:启动每个redis实例
5. Java中Jedis的使用
在Java中Jedis是对Redis服务的具体操作,可以直接将数据库中的数据通过开放的服务存储到Redis中。
5.1. Maven工程中配置依赖
需要把Jedis依赖的jar包添加到工程中。Maven工程中需要把jedis的坐标添加到依赖,配置pom.xml文件如下。
<!-- Redis客户端 -->
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.7.2</version>
</dependency>
5.2. 连接单个Redis服务
第一步:创建一个Jedis对象。需要指定服务端的ip及端口。
第二步:使用Jedis对象操作数据库,每个redis命令对应一个方法。
第三步:打印结果。
第四步:关闭Jedis
@Test
public void testJedis(){
//1.创建Jedis 对象 需要指定 连接的地址和端口
Jedis jedis = new Jedis("192.168.36.128", 6379);
//2.直接操作redis set
jedis.set("key1", "value1");
System.out.println(jedis.get("key1"));
//3.关闭Jedis
jedis.close();
}
5.3. 使用连接池连接单个Redis服务
第一步:创建一个JedisPool对象。需要指定服务端的ip及端口。
第二步:从JedisPool中获得Jedis对象。
第三步:使用Jedis操作Redis服务器。
第四步:操作完毕后关闭jedis对象,连接池回收资源。
第五步:关闭JedisPool对象。
@Test
public void teatJedisPool(){
//1.创建jedispool 对象需要指定端口和地址
JedisPool pool = new JedisPool("192.168.36.128", 6379);
//2.获取jedis的对象
Jedis jedis = pool.getResource();
//3.直接操作redis
jedis.set("keypool", "value");
System.out.println(jedis.get("keypool"));
//4.关闭redis (释放资源到连接池)
jedis.close();
//5.关闭连接池(应用系统关闭的时候才关闭)
pool.close();
}
5.4. 连接Redis集群
以六台服务器为例,操作步骤如下:
第一步:创建Set<HostAndPort>对象并添加ip和端口。这个作为Redis节点的列表。
第二步:创建JedisCluster对象,在系统中以单例的形式存在。
第三步:直接使用JedisCluster操作Redis服务器集群,操作与之前一致。
第三步:打印结果
第四步:关闭JedisCluster对象。
@Test
public void testjediscluster(){
Set<HostAndPort> nodes = new HashSet<>();
nodes.add(new HostAndPort("192.168.36.128", 7001));
nodes.add(new HostAndPort("192.168.36.136", 7002));
nodes.add(new HostAndPort("192.168.36.129", 7003));
nodes.add(new HostAndPort("192.168.36.130", 7004));
nodes.add(new HostAndPort("192.168.36.131", 7005));
nodes.add(new HostAndPort("192.168.36.132", 7006));
//1.创建jediscluster对象
JedisCluster cluster = new JedisCluster(nodes );
//2.直接根据jediscluster对象操作redis集群
cluster.set("keycluster", "cluster的value");
System.out.println(cluster.get("keycluster"));
//3.关闭jediscluster对象(是在应用系统关闭的时候关闭) 封装了连接池
cluster.close();
}
6. Java中SpringDataRedis的使用
6.1. Maven工程中配置依赖
要使用SpringDataRedis框架需加入Spring其他相关依赖以及Jedis相关依赖,为了方便测试,同时引入Junit依赖包,具体引用如下:
<properties>
<junit.version>4.12</junit.version>
<spring.version>4.2.4.RELEASE</spring.version>
</properties>
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>${junit.version}</version>
</dependency>
<!-- Spring -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-beans</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-jdbc</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jms</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-test</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- 缓存 -->
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.9.0</version>
</dependency>
<dependency>
<groupId>org.springframework.data</groupId>
<artifactId>spring-data-redis</artifactId>
<version>1.8.6.RELEASE</version>
</dependency>
</dependencies>
6.2. 添加配置文件
第一步:新建SpringDataRedis相关的配置文件,applicationContext-redis.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"
xmlns:cache="http://www.springframework.org/schema/cache"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/cache
http://www.springframework.org/schema/cache/spring-cache.xsd">
<context:property-placeholder location="classpath*:properties/*.properties" />
<!-- redis 相关配置 -->
<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>
<bean id="JedisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"
p:host-name="${redis.host}" p:port="${redis.port}" p:password="${redis.pass}" p:pool-config-ref="poolConfig"/>
<bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate">
<property name="connectionFactory" ref="JedisConnectionFactory" />
</bean>
</beans>
第二步:新建Redis连接配置文件,redis-config.properties。配置项如下:
# Redis settings
# server IP
redis.host=192.168.25.129
# server port
redis.port=6379
# server pass
redis.pass=
# use dbIndex
redis.database=0
redis.maxIdle=300
redis.maxWait=3000
redis.testOnBorrow=true
6.3. 引入RedisTemplate测试调用
新建测试脚本,使用@Autowired注解,引入RedisTemplate类,操作方法,进行Redis操作。
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:spring/applicationContext-redis.xml")
public class SpringDataRedisTest {
@Autowired
private RedisTemplate redisTemplate;
@Test
public void setValue(){
redisTemplate.boundValueOps("name").set("测试值");
}
@Test
public void getValue(){
String str = (String) redisTemplate.boundValueOps("name").get();
System.out.println(str);
}
@Test
public void deleteValue(){
redisTemplate.delete("name");;
}
}
7.扩展知识
7.1. 配置Mybatis二级缓存
示例项目为SpringBoot项目
7.1.1 配置文件修改
配置Redis数据库连接和打开Mybatis二级缓存。
spring:
redis:
lettuce:
pool:
max-active: 8
max-idle: 8
max-wait: -1ms
min-idle: 0
sentinel:
master: mymaster
nodes: 192.168.25.132:26379, 192.168.25.132:26380, 192.168.25.132:26381
mybatis:
configuration:
cache-enabled: true
7.1.2 实体类实现序列化接口并声明序列号
在所有的实体类中继承接口Serializable
,并什么序列号,序列号如下:
private static final long serialVersionUID = 546523264786598665L;
IDEA 提示生成序列号
默认情况下 Intellij IDEA 不会提示继承了 Serializable 接口的类生成 serialVersionUID 的警告。如果需要生成 serialVersionUID,需要手动配置。
File -> Settings -> Inspections -> Serialization issues -> Serialization class without ‘serialVersionUID’
7.1.3 实现 Spring ApplicationContextAware 接口,用于手动注入 Bean
创建一个名为 ApplicationContextHolder 的工具类,代码如下:
import org.apache.commons.lang3.Validate;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
@Component
public class ApplicationContextHolder implements ApplicationContextAware, DisposableBean {
private static final Logger logger = LoggerFactory.getLogger(ApplicationContextHolder.class);
private static ApplicationContext applicationContext;
/**
* 获取存储在静态变量中的 ApplicationContext
*
* @return
*/
public static ApplicationContext getApplicationContext() {
assertContextInjected();
return applicationContext;
}
/**
* 从静态变量 applicationContext 中获取 Bean,自动转型成所赋值对象的类型
*
* @param name
* @param <T>
* @return
*/
public static <T> T getBean(String name) {
assertContextInjected();
return (T) applicationContext.getBean(name);
}
/**
* 从静态变量 applicationContext 中获取 Bean,自动转型成所赋值对象的类型
*
* @param clazz
* @param <T>
* @return
*/
public static <T> T getBean(Class<T> clazz) {
assertContextInjected();
return applicationContext.getBean(clazz);
}
/**
* 实现 DisposableBean 接口,在 Context 关闭时清理静态变量
*
* @throws Exception
*/
public void destroy() throws Exception {
logger.debug("清除 SpringContext 中的 ApplicationContext: {}", applicationContext);
applicationContext = null;
}
/**
* 实现 ApplicationContextAware 接口,注入 Context 到静态变量中
*
* @param applicationContext
* @throws BeansException
*/
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
ApplicationContextHolder.applicationContext = applicationContext;
}
/**
* 断言 Context 已经注入
*/
private static void assertContextInjected() {
Validate.validState(applicationContext != null, "applicationContext 属性未注入,请在 spring-context.xml 配置中定义 ApplicationContextHolder");
}
}
7.1.4 实现 MyBatis Cache 接口,自定义缓存为 Redis
在项目中创建一个名为 RedisCache 的工具类,代码如下:
import com.funtl.myshop.commons.context.ApplicationContextHolder;
import org.apache.ibatis.cache.Cache;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.data.redis.core.RedisCallback;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.ValueOperations;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
/**
* Redis 缓存工具类
* <p>Title: RedisCache</p>
* <p>Description: </p>
*
* @version 1.0.0
*/
public class RedisCache implements Cache {
private static final Logger logger = LoggerFactory.getLogger(RedisCache.class);
private final ReadWriteLock readWriteLock = new ReentrantReadWriteLock();
private final String id; // cache instance id
private RedisTemplate redisTemplate;
private static final long EXPIRE_TIME_IN_MINUTES = 30; // redis过期时间
public RedisCache(String id) {
if (id == null) {
throw new IllegalArgumentException("Cache instances require an ID");
}
this.id = id;
}
@Override
public String getId() {
return id;
}
/**
* Put query result to redis
*
* @param key
* @param value
*/
@Override
public void putObject(Object key, Object value) {
try {
RedisTemplate redisTemplate = getRedisTemplate();
ValueOperations opsForValue = redisTemplate.opsForValue();
opsForValue.set(key, value, EXPIRE_TIME_IN_MINUTES, TimeUnit.MINUTES);
logger.debug("Put query result to redis");
} catch (Throwable t) {
logger.error("Redis put failed", t);
}
}
/**
* Get cached query result from redis
*
* @param key
* @return
*/
@Override
public Object getObject(Object key) {
try {
RedisTemplate redisTemplate = getRedisTemplate();
ValueOperations opsForValue = redisTemplate.opsForValue();
logger.debug("Get cached query result from redis");
// System.out.println("****" + opsForValue.get(key).toString());
return opsForValue.get(key);
} catch (Throwable t) {
logger.error("Redis get failed, fail over to db", t);
return null;
}
}
/**
* Remove cached query result from redis
*
* @param key
* @return
*/
@Override
@SuppressWarnings("unchecked")
public Object removeObject(Object key) {
try {
RedisTemplate redisTemplate = getRedisTemplate();
redisTemplate.delete(key);
logger.debug("Remove cached query result from redis");
} catch (Throwable t) {
logger.error("Redis remove failed", t);
}
return null;
}
/**
* Clears this cache instance
*/
@Override
public void clear() {
RedisTemplate redisTemplate = getRedisTemplate();
redisTemplate.execute((RedisCallback) connection -> {
connection.flushDb();
return null;
});
logger.debug("Clear all the cached query result from redis");
}
/**
* This method is not used
*
* @return
*/
@Override
public int getSize() {
return 0;
}
@Override
public ReadWriteLock getReadWriteLock() {
return readWriteLock;
}
private RedisTemplate getRedisTemplate() {
if (redisTemplate == null) {
redisTemplate = ApplicationContextHolder.getBean("redisTemplate");
}
return redisTemplate;
}
}
7.1.5 Mapper 接口中增加注解
在 Mapper 接口中增加 @CacheNamespace(implementation = RedisCache.class) 注解,声明需要使用二级缓存。
import com.funtl.myshop.commons.domain.TbUser;
import com.funtl.myshop.commons.utils.RedisCache;
import org.apache.ibatis.annotations.CacheNamespace;
import tk.mybatis.mapper.MyMapper;
@CacheNamespace(implementation = RedisCache.class)
public interface TbUserMapper extends MyMapper<TbUser> {}
文档下载地址:
https://wenku.baidu.com/view/30cd862bdc88d0d233d4b14e852458fb770b38b7
网友评论