9、redis之事务2-Jedis的八种调用方式(事务、管道、分布式)介绍
1、普通同步
- @Test
- public void test1Normal() {
- Jedis jedis = new Jedis("localhost");
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- String result = jedis.set("n" + i, "n" + i);
- }
- long end = System.currentTimeMillis();
- System.out.println("Simple SET: " + ((end - start)/1000.0) + " seconds");
- jedis.disconnect();
- }
例子中的for循环里的每一个set操作都是一个事务。
2、事务方式(Transactions)
redis的事务很简单,他主要目的是保障,一个client发起的事务中的命令可以连续的执行,而中间不会插入其他client的命令。
- @Test
- public void test2Trans() {
- Jedis jedis = new Jedis("localhost");
- long start = System.currentTimeMillis();
- Transaction tx = jedis.multi();
- for (int i = 0; i < 100000; i++) {
- tx.set("t" + i, "t" + i);
- }
- List<Object> results = tx.exec();
- long end = System.currentTimeMillis();
- System.out.println("Transaction SET: " + ((end - start)/1000.0) + " seconds");
- jedis.disconnect();
- }
我们调用jedis.watch(keys)
方法来监控key,如果调用后key值发生变化,则整个事务会执行失败。另外,事务中某个操作失败,并不会回滚其他操作。这一点需要注意。还有,我们可以使用discard()
方法来取消事务。
3、管道(Pipelining)
有时,我们需要采用异步方式,一次发送多个指令,不同步等待其返回结果。这样可以取得非常好的执行效率。这就是管道,调用方法如下:
- @Test
- public void test3Pipelined() {
- Jedis jedis = new Jedis("localhost");
- Pipeline pipeline = jedis.pipelined();
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- pipeline.set("p" + i, "p" + i);
- }
- List<Object> results = pipeline.syncAndReturnAll();
- long end = System.currentTimeMillis();
- System.out.println("Pipelined SET: " + ((end - start)/1000.0) + " seconds");
- jedis.disconnect();
- }
4、管道中调用事务
就Jedis提供的方法而言,是可以做到在管道中使用事务,其代码如下:
- @Test
- public void test4combPipelineTrans() {
- jedis = new Jedis("localhost");
- long start = System.currentTimeMillis();
- Pipeline pipeline = jedis.pipelined();
- pipeline.multi();
- for (int i = 0; i < 100000; i++) {
- pipeline.set("" + i, "" + i);
- }
- pipeline.exec();
- List<Object> results = pipeline.syncAndReturnAll();
- long end = System.currentTimeMillis();
- System.out.println("Pipelined transaction: " + ((end - start)/1000.0) + " seconds");
- jedis.disconnect();
- }
但是经测试(见本文后续部分),发现其效率和单独使用事务差不多,甚至还略微差点。
5、分布式直连同步调用
- @Test
- public void test5shardNormal() {
- List<JedisShardInfo> shards = Arrays.asList(
- new JedisShardInfo("localhost",6379),
- new JedisShardInfo("localhost",6380));
- ShardedJedis sharding = new ShardedJedis(shards);
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- String result = sharding.set("sn" + i, "n" + i);
- }
- long end = System.currentTimeMillis();
- System.out.println("Simple@Sharing SET: " + ((end - start)/1000.0) + " seconds");
- sharding.disconnect();
- }
这个是分布式直接连接,并且是同步调用,每步执行都返回执行结果。类似地,还有异步管道调用
6、分布式直连异步调用
- @Test
- public void test6shardpipelined() {
- List<JedisShardInfo> shards = Arrays.asList(
- new JedisShardInfo("localhost",6379),
- new JedisShardInfo("localhost",6380));
- ShardedJedis sharding = new ShardedJedis(shards);
- ShardedJedisPipeline pipeline = sharding.pipelined();
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- pipeline.set("sp" + i, "p" + i);
- }
- List<Object> results = pipeline.syncAndReturnAll();
- long end = System.currentTimeMillis();
- System.out.println("Pipelined@Sharing SET: " + ((end - start)/1000.0) + " seconds");
- sharding.disconnect();
- }
7、分布式连接池同步调用
如果,你的分布式调用代码是运行在线程中,那么上面两个直连调用方式就不合适了,因为直连方式是非线程安全的,这个时候,你就必须选择连接池调用
- @Test
- public void test7shardSimplePool() {
- List<JedisShardInfo> shards = Arrays.asList(
- new JedisShardInfo("localhost",6379),
- new JedisShardInfo("localhost",6380));
- ShardedJedisPool pool = new ShardedJedisPool(new JedisPoolConfig(), shards);
- ShardedJedis one = pool.getResource();
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- String result = one.set("spn" + i, "n" + i);
- }
- long end = System.currentTimeMillis();
- pool.returnResource(one);
- System.out.println("Simple@Pool SET: " + ((end - start)/1000.0) + " seconds");
- pool.destroy();
- }
上面是同步方式,当然还有异步方式
8、分布式连接池异步调用
- @Test
- public void test8shardPipelinedPool() {
- List<JedisShardInfo> shards = Arrays.asList(
- new JedisShardInfo("localhost",6379),
- new JedisShardInfo("localhost",6380));
- ShardedJedisPool pool = new ShardedJedisPool(new JedisPoolConfig(), shards);
- ShardedJedis one = pool.getResource();
- ShardedJedisPipeline pipeline = one.pipelined();
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- pipeline.set("sppn" + i, "n" + i);
- }
- List<Object> results = pipeline.syncAndReturnAll();
- long end = System.currentTimeMillis();
- pool.returnResource(one);
- System.out.println("Pipelined@Pool SET: " + ((end - start)/1000.0) + " seconds");
- pool.destroy();
- }
需要注意的地方:
1)事务和管道都是异步模式。在事务和管道中不能同步查询结果。比如下面两个调用,都是不允许的,如下:
- Transaction tx = jedis.multi();
- for (int i = 0; i < 100000; i++) {
- tx.set("t" + i, "t" + i);
- }
- System.out.println(tx.get("t1000").get()); //不允许
- List<Object> results = tx.exec();
- …
- …
- Pipeline pipeline = jedis.pipelined();
- long start = System.currentTimeMillis();
- for (int i = 0; i < 100000; i++) {
- pipeline.set("p" + i, "p" + i);
- }
- System.out.println(pipeline.get("p1000").get()); //不允许
- List<Object> results = pipeline.syncAndReturnAll();
2)事务和管道都是异步的,个人感觉,在管道中再进行事务调用,没有必要,不如直接进行事务模式。
3)分布式中,连接池的性能比直连的性能略好(见后续测试部分)。
4)分布式调用中不支持事务,因为事务是在服务器端实现,而在分布式中,每批次的调用对象都可能访问不同的机器,所以,没法进行事务。
转载自:http://www.open-open.com/lib/view/open1410485827242.html
9、redis之事务2-Jedis的八种调用方式(事务、管道、分布式)介绍的更多相关文章
- 【转载】Redis的Java客户端Jedis的八种调用方式(事务、管道、分布式…)介绍
转载地址:http://blog.csdn.net/truong/article/details/46711045 关键字:Redis的Java客户端Jedis的八种调用方式(事务.管道.分布式…)介 ...
- Redis的Java客户端Jedis的八种调用方式(事务、管道、分布式)介绍
jedis是一个著名的key-value存储系统,而作为其官方推荐的java版客户端jedis也非常强大和稳定,支持事务.管道及有jedis自身实现的分布式. 在这里对jedis关于事务.管道和分布式 ...
- Jedis的八种调用方式(功能:事务,管道)
1. packagecom.irwin.redis; 2. 3. importjava.util.Arrays; 4. importjava.util.List; ...
- Java客户端Jedis的八种调用方式
redis是一个著名的key-value存储系统,而作为其官方推荐的java版客户端jedis也非常强大和稳定,支持事务.管道及有jedis自身实现的分布式. 在这里对jedis关于事务.管道和分 ...
- Spring事务Transaction配置的五种注入方式详解
Spring事务Transaction配置的五种注入方式详解 前段时间对Spring的事务配置做了比较深入的研究,在此之间对Spring的事务配置虽说也配置过,但是一直没有一个清楚的认识.通过这次的学 ...
- Spring事务之详解--三种实现方式
实现购买股票案例: 一.引入JAR文件: 二.开始搭建分层架构---创建账户(Account)和股票(Stock)实体类 Account: /* * 账户 */ public class Accoun ...
- Selenium Webdriver元素定位的八种常用方式
楼主原创,欢迎学习和交流,码字不容易,转载请注明出处,谢谢. 在使用selenium webdriver进行元素定位时,通常使用findElement或findElements方法结合By类返回的元素 ...
- Selenium Webdriver元素定位的八种常用方式(转载)
转载自 https://www.cnblogs.com/qingchunjun/p/4208159.html 在使用selenium webdriver进行元素定位时,通常使用findElement或 ...
- 爬虫-【selenium—Webdriver元素定位的八种常用方式
在使用selenium webdriver进行元素定位时,通常使用findElement或findElements方法结合By类返回的元素句柄来定位元素.其中By类的常用定位方式共八种,现分别介绍如下 ...
随机推荐
- Orchard模块开发全接触2:新建 ProductPart
一:创建 Part 1:项目引用 Orchard.Framework: 2:创建 Models 文件夹: 3:在 Models 文件夹下创建类 ProductPartRecord,如下: public ...
- 在LaTeX中使用颜色 Using colours in LaTeX
Using colours in LaTeX There are several elements in LATEX whose colour can be changed to improve th ...
- 图像质量评估(IQA)
图像质量评估函数的分类曾是一个比较有争议的话题,在2l世纪以前曾经有过 比较多的讨论.但是随着研究的深入和技术的广泛应用,研究人员对于图像质量 评估函数的分类有了统一的认识,即从实际应用中参考信息供给 ...
- Installing Hyperledger Fabric v1.1 on Ubuntu 16.04 — Part II & Part III
This entire tutorial is the second part of the installation of Hyperledger Fabric v1.1. In the previ ...
- ECC校验
ECC的全称是 Error Checking and Correction or Error correction Coding,是一种用于差错检测和修正的算法.NAND闪存在生产和使用中都会产生坏块 ...
- golang常用模块介绍
golang模块 一.命令行库Cobra Cobra提供简单的接口来创建强大的现代化CLI接口,比如git与go工具.Cobra同时也是一个程序, 用于创建CLI程序 https://www.jian ...
- jQuery中attr()和prop()的区别,修改checked属性 jquery attr('checked' 不起作用 attr('checked' 不对
在做复选框全选按钮的时候,出现了一个问题,使用语句$.attr('checked',true),将复选框的属性改为被选中,在chrome浏览器中第一次点击有效后面就不行了,IE8倒是没有问题. 百度了 ...
- 【BZOJ】【3339】Rmq Problem
离线+线段树 Orz Hzwer,引用题解: 这一题在线似乎比较麻烦 至于离线.. 首先按照左端点将询问排序 然后一般可以这样考虑 首先如何得到1-i的sg值呢 这个可以一开始扫一遍完成 接着考虑l- ...
- 生成学习算法(Generative Learning algorithms)
一.引言 前面我们谈论到的算法都是在给定\(x\)的情况下直接对\(p(y|x;\theta)\)进行建模.例如,逻辑回归利用\(h_\theta(x)=g(\theta^T x)\)对\(p(y|x ...
- 7.5 zookeeper客户端curator的基本使用 + zkui
使用zookeeper原生API实现一些复杂的东西比较麻烦.所以,出现了两款比较好的开源客户端,对zookeeper的原生API进行了包装:zkClient和curator.后者是Netflix出版的 ...