一:环境准备
1. 引入依赖
<!--引入jedis连接依赖--><dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId><version>2.9.0</version></dependency>
2.创建jedis对象
public static void main(String[] args) {//1.创建jedis对象Jedis jedis = new Jedis("192.168.40.4", 6379);//1.redis服务必须关闭防火墙 2.redis服务必须开启远程连接jedis.select(0);//选择操作的库默认0号库//2.执行相关操作//....//3.释放资源jedis.close();}
1.1:操作key相关API
private Jedis jedis;@Beforepublic void before(){this.jedis = new Jedis("192.168.202.205", 7000);}@Afterpublic void after(){jedis.close();}//测试key相关@Testpublic void testKeys(){//删除一个keyjedis.del("name");//删除多个keyjedis.del("name","age");//判断一个key是否存在exitsBoolean name = jedis.exists("name");System.out.println(name);//设置一个key超时时间 expire pexpireLong age = jedis.expire("age", 100);System.out.println(age);//获取一个key超时时间 ttlLong age1 = jedis.ttl("newage");System.out.println(age1);//随机获取一个keyString s = jedis.randomKey();//修改key名称jedis.rename("age","newage");//查看可以对应值的类型String name1 = jedis.type("name");System.out.println(name1);String maps = jedis.type("maps");System.out.println(maps);}

1.2:操作String相关API
//测试String相关@Testpublic void testString(){//setjedis.set("name","小陈");//getString s = jedis.get("name");System.out.println(s);//msetjedis.mset("content","好人","address","海淀区");//mgetList<String> mget = jedis.mget("name", "content", "address");mget.forEach(v-> System.out.println("v = " + v));//getsetString set = jedis.getSet("name", "小明");System.out.println(set);//............}

1.3:操作List相关API
//测试List相关@Testpublic void testList(){//lpushjedis.lpush("names1","张三","王五","赵柳","win7");//rpushjedis.rpush("names1","xiaomingming");//lrangeList<String> names1 = jedis.lrange("names1", 0, -1);names1.forEach(name-> System.out.println("name = " + name));//lpop rpopString names11 = jedis.lpop("names1");System.out.println(names11);//llenjedis.linsert("lists", BinaryClient.LIST_POSITION.BEFORE,"xiaohei","xiaobai");//........}

1.4:操作Set的相关API
//测试SET相关@Testpublic void testSet(){//saddjedis.sadd("names","zhangsan","lisi");//smembersjedis.smembers("names");//sismemberjedis.sismember("names","xiaochen");//...}

1.5: 操作ZSet相关API
//测试ZSET相关@Testpublic void testZset(){//zaddjedis.zadd("names",10,"张三");//zrangejedis.zrange("names",0,-1);//zcardjedis.zcard("names");//zrangeByScorejedis.zrangeByScore("names","0","100",0,5);//..}

1.6: 操作Hash相关API
//测试HASH相关@Testpublic void testHash(){//hsetjedis.hset("maps","name","zhangsan");//hgetjedis.hget("maps","name");//hgetalljedis.hgetAll("mps");//hkeysjedis.hkeys("maps");//hvalsjedis.hvals("maps");//....}
二:SpringBoot整合Redis
Spring Boot Data(数据) Redis 中提供了RedisTemplate和StringRedisTemplate,其中StringRedisTemplate是RedisTemplate的子类,两个方法基本一致,不同之处主要体现在操作的数据类型不同,RedisTemplate中的两个泛型都是Object,意味着存储的key和value都可以是一个对象,而StringRedisTemplate的两个泛型都是String,意味着StringRedisTemplate的key和value都只能是字符串。
**注意: 使用RedisTemplate默认是将对象序列化到Redis中,所以放入的对象必须实现对象序列化接口
2:环境准备
2.1.引入依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency>
2.2.配置application.propertie
spring.redis.host=localhostspring.redis.port=6379spring.redis.database=0
3: 使用StringRedisTemplate和RedisTemplate
@Autowiredprivate StringRedisTemplate stringRedisTemplate; //对字符串支持比较友好,不能存储对象@Autowiredprivate RedisTemplate redisTemplate; //存储对象@Testpublic void testRedisTemplate(){System.out.println(redisTemplate);//设置redistemplate值使用对象序列化策略redisTemplate.setValueSerializer(new JdkSerializationRedisSerializer());//指定值使用对象序列化//redisTemplate.opsForValue().set("user",new User("21","小黑",23,new Date()));User user = (User) redisTemplate.opsForValue().get("user");System.out.println(user);// Set keys = redisTemplate.keys("*");// keys.forEach(key -> System.out.println(key));/*Object name = redisTemplate.opsForValue().get("name");System.out.println(name);*///Object xiaohei = redisTemplate.opsForValue().get("xiaohei");//System.out.println(xiaohei);/*redisTemplate.opsForValue().set("name","xxxx");Object name = redisTemplate.opsForValue().get("name");System.out.println(name);*//*redisTemplate.opsForList().leftPushAll("lists","xxxx","1111");List lists = redisTemplate.opsForList().range("lists", 0, -1);lists.forEach(list-> System.out.println(list));*/}//key的绑定操作 如果日后对某一个key的操作及其频繁,可以将这个key绑定到对应redistemplate中,日后基于绑定操作都是操作这个key//boundValueOps 用来对String值绑定key//boundListOps 用来对List值绑定key//boundSetOps 用来对Set值绑定key//boundZsetOps 用来对Zset值绑定key//boundHashOps 用来对Hash值绑定key@Testpublic void testBoundKey(){BoundValueOperations<String, String> nameValueOperations = stringRedisTemplate.boundValueOps("name");nameValueOperations.set("1");//yuewnameValueOperations.set("2");String s = nameValueOperations.get();System.out.println(s);}//hash相关操作 opsForHash@Testpublic void testHash(){stringRedisTemplate.opsForHash().put("maps","name","小黑");Object o = stringRedisTemplate.opsForHash().get("maps", "name");System.out.println(o);}//zset相关操作 opsForZSet@Testpublic void testZSet(){stringRedisTemplate.opsForZSet().add("zsets","小黑",10);Set<String> zsets = stringRedisTemplate.opsForZSet().range("zsets", 0, -1);zsets.forEach(value-> System.out.println(value));}//set相关操作 opsForSet@Testpublic void testSet(){stringRedisTemplate.opsForSet().add("sets","xiaosan","xiaosi","xiaowu");Set<String> sets = stringRedisTemplate.opsForSet().members("sets");sets.forEach(value-> System.out.println(value));}//list相关的操作opsForList@Testpublic void testList(){// stringRedisTemplate.opsForList().leftPushAll("lists","张三","李四","王五");List<String> lists = stringRedisTemplate.opsForList().range("lists", 0, -1);lists.forEach(key -> System.out.println(key));}//String相关的操作 opsForValue@Testpublic void testString(){//stringRedisTemplate.opsForValue().set("166","好同学");String s = stringRedisTemplate.opsForValue().get("166");System.out.println(s);Long size = stringRedisTemplate.opsForValue().size("166");System.out.println(size);}//key相关的操作@Testpublic void test(){Set<String> keys = stringRedisTemplate.keys("*");//查看所有keyBoolean name = stringRedisTemplate.hasKey("name");//判断某个key是否存在stringRedisTemplate.delete("age");//根据指定key删除stringRedisTemplate.rename("","");//修改key的名称stringRedisTemplate.expire("key",10, TimeUnit.HOURS);//设置key超时时间 参数1:设置key名 参数2:时间 参数3:时间的单位stringRedisTemplate.move("",1);//移动key}
4. Redis 主从复制
4.1 主从复制
主从复制架构仅仅用来解决数据的冗余备份,从节点仅仅用来同步数据
无法解决: 1.master节点出现故障的自动故障转移
4.2 主从复制架构图

4.3 搭建主从复制
# 1.准备3台机器并修改配置- masterport 6379bind 0.0.0.0- slave1port 6380bind 0.0.0.0slaveof masterip masterport- slave2port 6381bind 0.0.0.0slaveof masterip masterport

# 2.启动3台机器进行测试- cd /usr/redis/bin- ./redis-server /root/master/redis.conf- ./redis-server /root/slave1/redis.conf- ./redis-server /root/slave2/redis.conf
5:Redis哨兵机制
5.1 哨兵Sentinel机制
Sentinel(哨兵)是Redis 的高可用性解决方案:由一个或多个Sentinel 实例 组成的Sentinel 系统可以监视任意多个主服务器,以及这些主服务器属下的所有从服务器,并在被监视的主服务器进入下线状态时,自动将下线主服务器属下的某个从服务器升级为新的主服务器。简单的说哨兵就是带有自动故障转移功能的主从架构。
无法解决: 1.单节点并发压力问题 2.单节点内存和磁盘物理上限
5.2 哨兵架构原理

5.3 搭建哨兵架构
# 1.在主节点上创建哨兵配置- 在Master对应redis.conf同目录下新建sentinel.conf文件,名字绝对不能错;# 2.配置哨兵,在sentinel.conf文件中填入内容:- sentinel monitor 被监控数据库名字(自己起名字) ip port 1# 3.启动哨兵模式进行测试- redis-sentinel /root/sentinel/sentinel.conf说明:这个后面的数字2,是指当有两个及以上的sentinel服务检测到master宕机,才会去执行主从切换的功能。
5.4 通过springboot操作哨兵
# redis sentinel 配置# master书写是使用哨兵监听的那个名称spring.redis.sentinel.master=mymaster# 连接的不再是一个具体redis主机,书写的是多个哨兵节点spring.redis.sentinel.nodes=192.168.202.206:26379
- 注意:如果连接过程中出现如下错误:RedisConnectionException: DENIED Redis is running in protected mode because protected mode is enabled, no bind address was specified, no authentication password is requested to clients. In this mode connections are only accepted from the loopback interface. If you want to connect from external computers to Redis you may adopt one of the following solutions: 1) Just disable protected mode sending the command ‘CONFIG SET protected-mode no’ from the loopback interface by connecting to Redis from the same host the server is running, however MAKE SURE Redis is not publicly accessible from internet if you do so. Use CONFIG REWRITE to make this change permanent. 2)
- 解决方案:在哨兵的配置文件中加入bind 0.0.0.0 开启远程连接权限
三. Redis集群
3.1 集群
Redis在3.0后开始支持Cluster(模式)模式,目前redis的集群支持节点的自动发现,支持slave-master选举和容错,支持在线分片(sharding shard )等特性。reshard。
3.2 集群架构图
3.3 集群细节
- 所有的redis节点彼此互联(PING-PONG机制),内部使用二进制协议优化传输速度和带宽.
- 节点的fail是通过集群中超过半数的节点检测失效时才生效.
- 客户端与redis节点直连,不需要中间proxy层.客户端不需要连接集群所有节点,连接集群中任何一个可用节点即可
- redis-cluster把所有的物理节点映射到[0-16383]slot上,cluster 负责维护node<->slot<->value

3.4 集群搭建
判断一个是集群中的节点是否可用,是集群中的所用主节点选举过程,如果半数以上的节点认为当前节点挂掉,那么当前节点就是挂掉了,所以搭建redis集群时建议节点数最好为奇数,搭建集群至少需要三个主节点,三个从节点,至少需要6个节点。
# 1.准备环境安装ruby以及redis集群依赖- yum install -y ruby rubygems- gem install redis-xxx.gem

2.在一台机器创建7个目录

3.每个目录复制一份配置文件
# 3.每个目录复制一份配置文件[root@localhost ~]# cp redis-4.0.10/redis.conf 7000/[root@localhost ~]# cp redis-4.0.10/redis.conf 7001/[root@localhost ~]# cp redis-4.0.10/redis.conf 7002/[root@localhost ~]# cp redis-4.0.10/redis.conf 7003/[root@localhost ~]# cp redis-4.0.10/redis.conf 7004/[root@localhost ~]# cp redis-4.0.10/redis.conf 7005/[root@localhost ~]# cp redis-4.0.10/redis.conf 7006/

# 4.修改不同目录配置文件- port 6379 ..... //修改端口- bind 0.0.0.0 //开启远程连接- cluster-enabled yes //开启集群模式- cluster-config-file nodes-port.conf //集群节点配置文件- cluster-node-timeout 5000 //集群节点超时时间- appendonly yes //开启AOF持久化# 5.指定不同目录配置文件启动七个节点- [root@localhost bin]# ./redis-server /root/7000/redis.conf- [root@localhost bin]# ./redis-server /root/7001/redis.conf- [root@localhost bin]# ./redis-server /root/7002/redis.conf- [root@localhost bin]# ./redis-server /root/7003/redis.conf- [root@localhost bin]# ./redis-server /root/7004/redis.conf- [root@localhost bin]# ./redis-server /root/7005/redis.conf- [root@localhost bin]# ./redis-server /root/7006/redis.conf

# 6.查看进程- [root@localhost bin]# ps aux|grep redis
1.创建集群
# 1.复制集群操作脚本到bin目录中- [root@localhost bin]# cp /root/redis-4.0.10/src/redis-trib.rb .# 2.创建集群- ./redis-trib.rb create --replicas 1 192.168.202.205:7000 192.168.202.205:7001 192.168.202.205:7002 192.168.202.205:7003 192.168.202.205:7004 192.168.202.205:7005

3.集群创建成功出现如下提示

2.查看集群状态
# 1.查看集群状态 check [原始集群中任意节点] [无]- ./redis-trib.rb check 192.168.202.205:7000# 2.集群节点状态说明- 主节点主节点存在hash slots,且主节点的hash slots 没有交叉主节点不能删除一个主节点可以有多个从节点主节点宕机时多个副本之间自动选举主节点- 从节点从节点没有hash slots从节点可以删除从节点不负责数据的写,只负责数据的同步
3.添加主节点
# 1.添加主节点 add-node [新加入节点] [原始集群中任意节点]- ./redis-trib.rb add-node 192.168.1.158:7006 192.168.1.158:7005- 注意:1.该节点必须以集群模式启动2.默认情况下该节点就是以master节点形式添加
4.添加从节点
# 1.添加从节点 add-node --slave [新加入节点] [集群中任意节点]- ./redis-trib.rb add-node --slave 192.168.1.158:7006 192.168.1.158:7000- 注意:当添加副本节点时没有指定主节点,redis会随机给副本节点较少的主节点添加当前副本节点# 2.为确定的master节点添加主节点 add-node --slave --master-id master节点id [新加入节点] [集群任意节点]- ./redis-trib.rb add-node --slave --master-id 3c3a0c74aae0b56170ccb03a76b60cfe7dc1912e 127.0.0.1:7006 127.0.0.1:7000
5.删除副本节点
# 1.删除节点 del-node [集群中任意节点] [删除节点id]- ./redis-trib.rb del-node 127.0.0.1:7002 0ca3f102ecf0c888fc7a7ce43a13e9be9f6d3dd1- 注意:1.被删除的节点必须是从节点或没有被分配hash slots的节点
6.集群在线分片
# 1.在线分片 reshard [集群中任意节点] [无]- ./redis-trib.rb reshard 192.168.1.158:7000
四.Redis实现分布式Session管理
4.1 管理机制
redis的session管理是利用spring提供的session管理解决方案,将一个应用session交给Redis存储,整个应用中所有session的请求都会去redis中获取对应的session数据。
4.2 开发Session管理
1. 引入依赖
<dependency><groupId>org.springframework.session</groupId><artifactId>spring-session-data-redis</artifactId></dependency>
2. 开发Session管理配置类
@Configuration@EnableRedisHttpSessionpublic class RedisSessionManager {}


