Redis客户端开发包:Jedis学习-高级应用
事务
Jedis中事务的写法是将redis操作写在事物代码块中,如下所示,multi与exec之间为具体的事务。
jedis.watch (key1, key2, ...);
Transaction t = jedis.multi();
t.set("foo", "bar");
t.exec();
另外,在事务内部,是不能通过Jedis对象去获取值的,不过可以通过Transaction对象去获取,如下写法:
package cn.edu.hdu.jedisdemo; import java.util.Set; import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;
import redis.clients.jedis.Response;
import redis.clients.jedis.Transaction; public class App {
public static void main(String[] args) {
JedisPool pool = new JedisPool(new JedisPoolConfig(), "localhost");
// / Jedis implements Closable. Hence, the jedis instance will be
// auto-closed after the last statement.
try (Jedis jedis = pool.getResource()) {
// / ... do stuff here ... for example Transaction t = jedis.multi();
t.set("fool", "bar");
//jedis.get("fool");//报错,Cannot use Jedis when in Multi. Please use Transation or reset jedis state.
Response<String> result1 = t.get("fool"); t.zadd("foo", 1, "barowitch");
t.zadd("foo", 0, "barinsky");
t.zadd("foo", 0, "barikoviev");
Response<Set<String>> sose = t.zrange("foo", 0, -1);
t.exec(); // dont forget it String foolbar = result1.get(); // use Response.get() to retrieve things from a Response
int soseSize = sose.get().size(); // on sose.get() you can directly call Set methods! // List<Object> allResults = t.exec(); // you could still get all
// results at once, as before System.out.println(foolbar);
System.out.println(soseSize); }
// / ... when closing your application:
pool.close();
}
}
注意:Response对象的get方法要在事务exec方法执行之后调用,在t.exec()方法执行之前,Response对象是不包含结果的
管道
有时候,我们需要发送多个命令,一个高效的方法是使用管道;
通过管道可以一次性发送多条命令并在执行完后一次性将结果返回,当一组命令中每条命令都不依赖之前命令的执行结果时就可以将这组命令一起通过管道发出;
管道可以减少客户端与redis的通信次数,提供性能。
如下示例:
Pipeline p = jedis.pipelined();
p.set("fool", "bar");
p.zadd("foo", 1, "barowitch"); p.zadd("foo", 0, "barinsky"); p.zadd("foo", 0, "barikoviev");
Response<String> pipeString = p.get("fool");
Response<Set<String>> sose = p.zrange("foo", 0, -1);
p.sync(); int soseSize = sose.get().size();
Set<String> setBack = sose.get();
发布/订阅
订阅消息
先创建一个JedisPubSub对象,然后使用Jedis对象调用subscribe方法即可,该方法需要传入JedisPubSub对象;
MyListener listener = new MyListener(); //listener为JedisPubSub对象,监听类
jedis.subscribe(listener, "channel01");
发布消息
发布消息调用Jedis的publish方法即可;
jedis.publish(channel, message)
注意:subscribe方法是一个阻塞的操作,且发布和订阅不能使用同一个Jedis对象
一个完整的例子
监听事件类:
package cn.edu.hdu.jedisdemo;
import redis.clients.jedis.JedisPubSub;
class MyListener extends JedisPubSub {
public void onMessage(String channel, String message) {
System.out.println("get a msg: " + "channel=" + channel + ", message=" + message);
}
public void onSubscribe(String channel, int subscribedChannels) {
System.out.println("channel:" + channel + ", subscribedChannels:" + subscribedChannels);
}
public void onUnsubscribe(String channel, int subscribedChannels) {
System.out.println("channel:" + channel + ", subscribedChannels:" + subscribedChannels);
}
public void onPSubscribe(String pattern, int subscribedChannels) {
System.out.println("pattern:" + pattern + ", subscribedChannels:" + subscribedChannels);
}
public void onPUnsubscribe(String pattern, int subscribedChannels) {
System.out.println("pattern:" + pattern + ", subscribedChannels:" + subscribedChannels);
}
public void onPMessage(String pattern, String channel, String message) {
System.out.println("pattern:" + pattern + ", channel:" + channel + ", message:" + message);
}
}
main方法:
package cn.edu.hdu.jedisdemo; import java.util.Date;
import java.util.concurrent.TimeUnit; import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig; public class App {
public static void main(String[] args) {
final JedisPool pool = new JedisPool(new JedisPoolConfig(), "localhost"); //两个订阅者
new Thread(new Runnable() {
@Override
public void run() {
Jedis jedis = pool.getResource();
MyListener listener = new MyListener();
jedis.subscribe(listener, "channel01"); }
}).start(); new Thread(new Runnable() {
@Override
public void run() {
Jedis jedis = pool.getResource();
MyListener listener = new MyListener(); //listener为JedisPubSub对象,监听类
jedis.subscribe(listener, "channel01"); }
}).start(); //一个发布者
new Thread(new Runnable() {
@Override
public void run() {
//注意,订阅使用的jedis对象与发布使用的jedis对象不能相同,否者运行时会报错
Jedis jedis = pool.getResource();
while(true){
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
String msg = new Date().toLocaleString();
System.out.println("publish a msg:" + msg);
jedis.publish("channel01", msg);
} }
}).start(); //pool.close();
}
}
执行结果:
channel:channel01, subscribedChannels:1
channel:channel01, subscribedChannels:1
publish a msg:2016-7-20 10:03:42
get a msg: channel=channel01, message=2016-7-20 10:03:42
get a msg: channel=channel01, message=2016-7-20 10:03:42
publish a msg:2016-7-20 10:03:43
get a msg: channel=channel01, message=2016-7-20 10:03:43
get a msg: channel=channel01, message=2016-7-20 10:03:43
publish a msg:2016-7-20 10:03:44
get a msg: channel=channel01, message=2016-7-20 10:03:44
get a msg: channel=channel01, message=2016-7-20 10:03:44
分布式ShardedJedis
ShardedJedis里面采用了一致性哈希的算法,来决定每个key的保存位置,主要用于分摊服务器的压力。
举个简单的例子:
首先,开启三个redis服务器,分别使用不同的端口(拷贝三份,使用不同的配置文件即可):6379、6380、6381

然后,编写测试代码:
package cn.edu.hdu.jedisdemo; import java.util.Arrays;
import java.util.List;
import redis.clients.jedis.Client;
import redis.clients.jedis.JedisPoolConfig;
import redis.clients.jedis.JedisShardInfo;
import redis.clients.jedis.ShardedJedis;
import redis.clients.jedis.ShardedJedisPool; public class App {
public static void main(String[] args) {
// 设置连接池的相关配置
JedisPoolConfig poolConfig = new JedisPoolConfig();
poolConfig.setMaxTotal(2);
poolConfig.setMaxIdle(1);
poolConfig.setMaxWaitMillis(2000);
poolConfig.setTestOnBorrow(false);
poolConfig.setTestOnReturn(false); //设置Redis信息
String host = "127.0.0.1";
JedisShardInfo shardInfo1 = new JedisShardInfo(host, 6379, 500);
JedisShardInfo shardInfo2 = new JedisShardInfo(host, 6380, 500);
JedisShardInfo shardInfo3 = new JedisShardInfo(host, 6381, 500); //初始化ShardedJedisPool
List<JedisShardInfo> infoList = Arrays.asList(shardInfo1, shardInfo2, shardInfo3);
ShardedJedisPool jedisPool = new ShardedJedisPool(poolConfig, infoList); //一些基本操作
try(ShardedJedis jedis = jedisPool.getResource()) {
jedis.set("test", "test");
String test = jedis.get("test");
System.out.println(test);
} //查看
ShardedJedis jedis = jedisPool.getResource();
jedis.set("1111", "1111");
jedis.set("2222", "2222");
jedis.set("3333", "3333");
jedis.set("4444", "4444");
Client client1 = jedis.getShard("1111").getClient();
Client client2 = jedis.getShard("2222").getClient();
Client client3 = jedis.getShard("3333").getClient();
Client client4 = jedis.getShard("4444").getClient();
//打印key在哪个server中
System.out.println("1111 in server:" + client1.getHost() + " and port is:" + client1.getPort());
System.out.println("2222 in server:" + client2.getHost() + " and port is:" + client2.getPort());
System.out.println("3333 in server:" + client3.getHost() + " and port is:" + client3.getPort());
System.out.println("4444 in server:" + client4.getHost() + " and port is:" + client4.getPort());
jedis.close();
jedisPool.close();
}
}
查看输出结果:
test
1111 in server:127.0.0.1 and port is:6380
2222 in server:127.0.0.1 and port is:6379
3333 in server:127.0.0.1 and port is:6381
4444 in server:127.0.0.1 and port is:6381
参考内容:
https://github.com/xetorthio/jedis/wiki/AdvancedUsage
http://www.cnblogs.com/coder2012/p/4401153.html
http://blog.csdn.net/lihao21/article/details/48370687
http://www.cnblogs.com/vhua/p/redis_1.html
Redis客户端开发包:Jedis学习-高级应用的更多相关文章
- Redis客户端开发包:Jedis学习-入门
添加Jedis依赖 我们可以使用以下三种方式来添加Jedis依赖. 1.下载jar文件 从http://search.maven.org/下载最近的jedis包和Apache Commons Pool ...
- Tedis:淘宝的Redis的Java客户端开发包
Tedis:淘宝的Redis的Java客户端开发包 http://www.open-open.com/lib/view/open1389880631976.html Tedis Tedis是另 ...
- WebDav的java客户端开发包:Jackrabbit
上一篇帖子“WebDav的java客户端开发包:sardine”中说到,对于开发WebDav客户端 sardine是一个很好的选择,但sardine并未实现WevDav的全部规范,所以我又试了试 ap ...
- Redis客户端操作之Jedis
在前面的文章给大家详细的介绍了Redis各种知识点,本文主要给大家介绍下java程序怎么操作Redis数据. Redis之集群环境搭建 Redis官网中提供了各种语言的客户端,使用起来很方便, ...
- Java开发包Jedis
Jedis: http://www.oschina.net/p/jedis (Redis的官方首选Java开发包) <!--Redis --> <dependency> < ...
- WebDav的java客户端开发包:sardine
最近需要对WebDav服务器进行操作,查找了一下,基于java的开发包主要有这几个: slide Jackrabbit sardine webdavclient4j 其中slide是apache的一个 ...
- Redis客户端API操作 Jedis详解
redis是一个著名的key-value存储系统,也是nosql中的最常见的一种.其实,个人认为,redis最强大的地方不在于其存储,而在于其强大的缓存作用. 我们可以把它想象成一个巨大的(多借点集群 ...
- Redis客户端——Jedis的使用
本文介绍基于Java语言的Redis客户端——Jedis的使用,包括Jedis简介.获取Jedis.Jedis直连.Jedis连接池以及二者的对比的选择. Jedis简介 Jedis 是 Redis ...
- 深入剖析Redis客户端Jedis的特性和原理
一.开篇 Redis作为目前通用的缓存选型,因其高性能而倍受欢迎.Redis的2.x版本仅支持单机模式,从3.0版本开始引入集群模式. Redis的Java生态的客户端当中包含Jedis.Rediss ...
随机推荐
- SCC重新建图
Tarjan或Kosaraju算法[对每个点归类belong]求出SCC之后,对num_scc个SCC重新建图,针对不同问题,考虑重边的问题. //************************** ...
- XML EXtensible Markup Language
1.基础:XML设计被用来传输和存储数据:全称是EXtensible Markup Language.它的设计宗旨是传输数据,而不是显示数据.xml的标签没有被预定义,需要由用户自行定义标签.xml被 ...
- redis持久化机制
redis持久化 redis的数据存在内存中,所以存取性能好.但是存在内存中的数据存在一个问题,一旦机器重启,内存数据消失.为了解决这个问题,redis支持持久化.持久化就是为了解决内存数据丢失时恢复 ...
- Mac OS Git 安装
一.Git是一个分布式的代码版本管理工具.类似的常用工具还有SVN,CVS.最大的特点也是优点在于提供分布式的代码管理 1.分支代码只有一份! 使用过svn的童鞋想必都知道,当我们要开发一个新功能或者 ...
- GJM :自定义基于 VLC 的视频播放器 [转载]
感谢您的阅读.喜欢的.有用的就请大哥大嫂们高抬贵手"推荐一下"吧!你的精神支持是博主强大的写作动力以及转载收藏动力.欢迎转载! 版权声明:本文原创发表于 [请点击连接前往] ,未经 ...
- 学习hadoop遇到的问题
1.运行hadoop的帐号,需要多大的权限? 之前实验用的帐号是root,后来新建了个普通帐号,发觉不能直接套用到已经建立好的hadoop系统,因为无权限,即使将普通帐号加入到root群组仍然存在异常 ...
- Perfect Scrollbar – 完美的 jQuery 滚动条插件
Perfect Scrollbar 是一个很小的,但完美的 jQuery 滚动插件.滚动条不会影响原来的设计布局,滚动条的设计是完全可定制的.你可以改变几乎所有的 CSS 样式的滚动条,滚动条设计对脚 ...
- 使用 JavaScript 实现灵活的固定导航功能
如果你想在网页中实现灵活的固定导航功能,那么 Smart Fixed Navigation 这个 JavaScript 小脚本可以帮助轻松实现一个固定的导航,让用户在访问你的网站的时候可以随时使用菜单 ...
- HTML5 Maker – 在线轻松制作 HTML5 动画效果
HTML5 Maker 是一个在线动画制作工具,帮助你使用 HTML,CSS 和 JavaScript 创建动态,互动的内容.它非常容易使用,同时可以帮你实现非常好的效果.它可以制作跨浏览器的动画内容 ...
- 【position也可以很复杂】当弹出层遇上了鼠标定位(上)
前言 周五时同事有一个关于弹出层的问题没有解决,但是面临下班问题,我有点不舒服,便叫回去周六过来解决,但是上周六病了,所以请了个假,于是故事发生啦.... 今天上班时候,组员们卡到了那个地方,然后结果 ...