redis 学习笔记(7)-cluster 客户端(jedis)代码示例
上节学习了cluster的搭建及redis-cli终端下如何操作,但是更常用的场景是在程序代码里对cluster读写,这需要redis-client对cluster模式的支持,目前spring-data-redis(1.6.4)还不支持cluster,最新的1.7.0 RC1已经有cluster的相关实现了,不过目前尚未正式发布,所以现阶段要使用redis-cluster的话,client最好还是选用原生的jedis,示例代码如下:
配置文件:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="jedisCluster" class="redis.clients.jedis.JedisCluster">
<constructor-arg index="0">
<set>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="127.0.0.1"/>
<constructor-arg name="port" value="7000"/>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="127.0.0.1"/>
<constructor-arg name="port" value="7001"/>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="127.0.0.1"/>
<constructor-arg name="port" value="7002"/>
</bean>
<!--<bean class="redis.clients.jedis.HostAndPort">-->
<!--<constructor-arg name="host" value="127.0.0.1"/>-->
<!--<constructor-arg name="port" value="7003"/>-->
<!--</bean>-->
<!--<bean class="redis.clients.jedis.HostAndPort">-->
<!--<constructor-arg name="host" value="127.0.0.1"/>-->
<!--<constructor-arg name="port" value="7004"/>-->
<!--</bean>-->
<!--<bean class="redis.clients.jedis.HostAndPort">-->
<!--<constructor-arg name="host" value="127.0.0.1"/>-->
<!--<constructor-arg name="port" value="7005"/>-->
<!--</bean>-->
</set>
</constructor-arg>
</bean>
</beans>
注:上面的这些节点,不需要配全,最少可以只保留一个cluster中的节点信息,运行时,jedis会自动发现其它节点,但是为了防止某个节点挂掉,所以建议配置时,还是多配置几个,保证这一堆节点中,至少有一个能连接上。
示例代码:
package com.cnblogs.yjmyzz.redis; import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import redis.clients.jedis.JedisCluster;
import redis.clients.jedis.JedisPool; import java.util.List;
import java.util.Map;
import java.util.Set; public class AppDemo { private static Logger logger = LoggerFactory.getLogger(AppDemo.class);
private static JedisCluster jc = null; private static final String KEYS_STRING = "STRING";
private static final String KEYS_SET = "SET";
private static final String KEYS_LIST = "LIST";
private static final String KEYS_HASH = "HASH";
private static final String KEYS_ZSET = "ZSET"; private static void addKey(final String conainter, final String key) {
if (!jc.exists(conainter)) {
jc.sadd(conainter, key);
} else {
if (!jc.smembers(conainter).contains(key)) {
jc.sadd(conainter, key);
}
}
} /**
* 写入字符串缓存
*
* @param key
* @param value
* @return
*/
private static String set(final String key, final String value) {
String result = jc.set(key, value);
addKey(KEYS_STRING, key);
return result;
} /**
* 写入Set缓存
*
* @param key
* @param member
* @return
*/
private static Long sadd(final String key, final String... member) {
Long result = jc.sadd(key, member);
addKey(KEYS_SET, key);
return result;
} /**
* 从左侧写入List
*
* @param key
* @param string
* @return
*/
private static Long lpush(final String key, final String... string) {
Long result = jc.lpush(key, string);
addKey(KEYS_LIST, key);
return result;
} /**
* 写入HashMap缓存
*
* @param key
* @param field
* @param value
* @return
*/
private static Long hset(final String key, final String field, final String value) {
Long result = jc.hset(key, field, value);
addKey(KEYS_HASH, key);
return result;
} /**
* 写入ZSet缓存
*
* @param key
* @param score
* @param member
* @return
*/
private static Long zadd(final String key, final double score, final String member) {
Long result = jc.zadd(key, score, member);
addKey(KEYS_ZSET, key);
return result;
} private static Long zadd(final String key, final String member) {
Long result = jc.zadd(key, 0d, member);
addKey(KEYS_ZSET, key);
return result;
} public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-redis.xml"); jc = ctx.getBean(JedisCluster.class); Map<String, JedisPool> nodes = jc.getClusterNodes();
for (Map.Entry<String, JedisPool> entry : nodes.entrySet()) {
logger.info(entry.getKey() + " => " + entry.getValue().toString());
//清空所有数据
try {
entry.getValue().getResource().flushDB();
} catch (Exception e) {
logger.info(e.getLocalizedMessage());//slave节点上执行flushDB会报错
}
//entry.getValue().getResource().keys("*");//慎用,缓存数量较大时,会引起性能问题.
} //检测key是否存在
logger.info(jc.exists("a").toString()); //字符串写入测试
logger.info(set("a", "hello world!"));
logger.info(set("b", "hello redis!")); //字符串读取测试
logger.info(jc.get("a")); //set写入操作
logger.info("set写入测试 ==>");
logger.info(sadd("set1", "a", "b", "c") + ""); //缓存类型测试
logger.info(jc.type("set1")); //set读取测试
logger.info("set读取测试 ==>");
Set<String> set1 = jc.smembers("set1");
for (String s : set1) {
logger.info(s);
} //list写入测试
logger.info("list写入测试 ==>");
logger.info(lpush("list1", "1", "2", "3") + ""); //list读取测试
logger.info("list读取测试 ==>");
List<String> list1 = jc.lrange("list1", 0, 999);
for (String s : list1) {
logger.info(s);
} //hash写入测试
logger.info("hash写入测试 ==>");
logger.info(hset("hash1", "jimmy", "杨俊明") + "");
logger.info(hset("hash1", "CN", "中国") + "");
logger.info(hset("hash1", "US", "美国") + ""); //hash读取测试
logger.info("hash读取测试 ==>");
Map<String, String> hash1 = jc.hgetAll("hash1");
for (Map.Entry<String, String> entry : hash1.entrySet()) {
logger.info(entry.getKey() + ":" + entry.getValue());
} //zset写入测试
logger.info("zset写入测试 ==>");
logger.info(zadd("zset1", "3") + "");
logger.info(zadd("zset1", "2") + "");
logger.info(zadd("zset1", "1") + "");
logger.info(zadd("zset1", "4") + "");
logger.info(zadd("zset1", "5") + "");
logger.info(zadd("zset1", "6") + ""); //zset读取测试
logger.info("zset读取测试 ==>");
Set<String> zset1 = jc.zrange("zset1", 0, 999);
for (String s : zset1) {
logger.info(s);
} //遍历所有缓存项的key
logger.info("遍历cluster中的所有key ==>");
logger.info(jc.smembers(KEYS_STRING).toString());
logger.info(jc.smembers(KEYS_HASH).toString());
logger.info(jc.smembers(KEYS_SET).toString());
logger.info(jc.smembers(KEYS_LIST).toString());
logger.info(jc.smembers(KEYS_ZSET).toString()); }
}
注:建议尽量避免用jedis对节点做keys的模糊搜索,该操作在缓存项较多时,可能会导致redis性能急剧下降,改进办法是自己弄一个集合,记录所有缓存的key,具体可参考上面的办法。此外,jedis提供的命令非常之多,但是没有详细的说明文档(估计,作者认为代码就是最好的文档),大体可以从方法前缀猜测出来,比如sXXX表示是对Set的操作,hXXX表示是对hash的操作,lXXX或rXXX是对list的操作,zXXX是对zset的操作,什么前缀都没有的,比如set/get是对字符串的操作。
有网友把jedis的操作整理了一份文档,请参见:http://blog.csdn.net/zhu_xun/article/details/16806285
最后,附加上文中示例的源码:https://github.com/yjmyzz/redis-cluster-demo
redis 学习笔记(7)-cluster 客户端(jedis)代码示例的更多相关文章
- redis 学习笔记(6)-cluster集群搭建
上次写redis的学习笔记还是2014年,一转眼已经快2年过去了,在段时间里,redis最大的变化之一就是cluster功能的正式发布,以前要搞redis集群,得借助一致性hash来自己搞shardi ...
- redis学习笔记(二)——java中jedis的简单使用
redis怎么在java中使用,那就是要用到jedis了,jedis是redis的java版本的客户端实现,原本原本想上来就直接学spring整合redis的,但是一口吃个胖子,还是脚踏实地,从基础开 ...
- redis 学习笔记(5)-Spring与Jedis的集成
首先不得不服Spring这个宇宙无敌的开源框架,几乎整合了所有流行的其它框架,http://projects.spring.io/spring-data/从这上面看,当下流行的redis.solr.h ...
- Redis学习笔记(4)—— Jedis入门
一.Jedis介绍 Redis不仅是使用命令来操作,现在基本上主流的语言都有客户端支持,比如Java.C.C#.C++.php.Node.js.Go等. 在官方网站里列的一些Java客户端,有jedi ...
- redis 学习笔记-cluster集群搭建
一.下载最新版redis 编译 目前最新版是3.0.7,下载地址:http://www.redis.io/download 编译很简单,一个make命令即可,不清楚的同学,可参考我之前的笔记: red ...
- Redis学习笔记7--Redis管道(pipeline)
redis是一个cs模式的tcp server,使用和http类似的请求响应协议.一个client可以通过一个socket连接发起多个请求命令.每个请求命令发出后client通常会阻塞并等待redis ...
- (转)redis 学习笔记(1)-编译、启动、停止
redis 学习笔记(1)-编译.启动.停止 一.下载.编译 redis是以源码方式发行的,先下载源码,然后在linux下编译 1.1 http://www.redis.io/download 先 ...
- Redis 学习笔记4: Redis 3.2.1 集群搭建
在CenOS 6.7 linux环境下搭建Redis 集群环境 1.下载最新的Redis版本 本人下载的Redis版本是3.2.1版本,下载之后,解压,编译(make): 具体操作可以参考我的博文:R ...
- Redis学习笔记八:集群模式
作者:Grey 原文地址:Redis学习笔记八:集群模式 前面提到的Redis学习笔记七:主从复制和哨兵只能解决Redis的单点压力大和单点故障问题,接下来要讲的Redis Cluster模式,主要是 ...
随机推荐
- Vertica删除历史分区数据
假设test用户下创建的t_jingyu表 vsql -Utest -wtestpwd create table t_jingyu( col1 int, col2 varchar, col3 time ...
- java多线程--同步屏障CyclicBarrier的使用
CyclicBarrier的概念理解: CyclicBarrier的字面上的意思是可循环的屏障,是java并发包java.util.concurrent 里的一个同步工具类,在我下载的JDK1.6的中 ...
- 自己动手之使用反射和泛型,动态读取XML创建类实例并赋值
前言: 最近小匹夫参与的游戏项目到了需要读取数据的阶段了,那么觉得自己业余时间也该实践下数据相关的内容.那么从哪入手呢?因为用的是Unity3d的游戏引擎,思来想去就选择了C#读取XML文件这个小功能 ...
- JavaWeb——tomcat安装及目录介绍
一.web web可以说,就是一套 请求->处理->响应 的流程.客户端使用浏览器(IE.FireFox等),通过网络(Network)连接到服务器上,使用HTTP协议发起请求(Reque ...
- UWP简单示例(一):快速合成音乐MV
准备 IDE:Visual Studio 2015 为你的项目安装Nuget包 SharpDx.XAudio2 为你的项目安装Nuget包 Win2D.UWP 了解并学习:Win2D官方博客 了解并学 ...
- Java事务处理
Java事务处理总结 一.什么是Java事务 通常的观念认为,事务仅与数据库相关. 事务必须服从ISO/IEC所制定的ACID原则.ACID是原子性(atomicity).一致性(co ...
- CentOS6.8 修改主机名(1)
1.临时修改主机名 显示主机名:spark@master:~$ hostnamemaster修改主机名:spark@master:~$ sudo hostname hadoopspark@mast ...
- (转)配置Log4j(很详细)
来自:http://blog.csdn.net/yttcjj/article/details/37957317 Log4J的配置文件(Configuration File)就是用来设置记录器的级别.存 ...
- 怎样给div增加resize事件
当浏览器窗口被调整到一个新的高度或宽度时,就会触发resize事件,这个事件在window上面触发,那么如何给div元素增加resize事件,监听div的高度或宽度的改变呢? 先来回答另一个问题,监听 ...
- SVG 文本
该部分为四个主要部分: 1. <text>和<tspan>标签详解 2. 文本水平垂直居中问题 3. <textpath>让文本在指定路径上排列 4 ...