1、普通同步

  1. @Test
  2. public void test1Normal() {
  3. Jedis jedis = new Jedis("localhost");
  4. long start = System.currentTimeMillis();
  5. for (int i = 0; i < 100000; i++) {
  6. String result = jedis.set("n" + i, "n" + i);
  7. }
  8. long end = System.currentTimeMillis();
  9. System.out.println("Simple SET: " + ((end - start)/1000.0) + " seconds");
  10. jedis.disconnect();
  11. }

例子中的for循环里的每一个set操作都是一个事务。

2、事务方式(Transactions)

redis的事务很简单,他主要目的是保障,一个client发起的事务中的命令可以连续的执行,而中间不会插入其他client的命令。

  1. @Test
  2. public void test2Trans() {
  3. Jedis jedis = new Jedis("localhost");
  4. long start = System.currentTimeMillis();
  5. Transaction tx = jedis.multi();
  6. for (int i = 0; i < 100000; i++) {
  7. tx.set("t" + i, "t" + i);
  8. }
  9. List<Object> results = tx.exec();
  10. long end = System.currentTimeMillis();
  11. System.out.println("Transaction SET: " + ((end - start)/1000.0) + " seconds");
  12. jedis.disconnect();
  13. }

我们调用jedis.watch(keys)方法来监控key,如果调用后key值发生变化,则整个事务会执行失败。另外,事务中某个操作失败,并不会回滚其他操作。这一点需要注意。还有,我们可以使用discard()方法来取消事务。

3、管道(Pipelining)

有时,我们需要采用异步方式,一次发送多个指令,不同步等待其返回结果。这样可以取得非常好的执行效率。这就是管道,调用方法如下:

  1. @Test
  2. public void test3Pipelined() {
  3. Jedis jedis = new Jedis("localhost");
  4. Pipeline pipeline = jedis.pipelined();
  5. long start = System.currentTimeMillis();
  6. for (int i = 0; i < 100000; i++) {
  7. pipeline.set("p" + i, "p" + i);
  8. }
  9. List<Object> results = pipeline.syncAndReturnAll();
  10. long end = System.currentTimeMillis();
  11. System.out.println("Pipelined SET: " + ((end - start)/1000.0) + " seconds");
  12. jedis.disconnect();
  13. }

4、管道中调用事务

就Jedis提供的方法而言,是可以做到在管道中使用事务,其代码如下:

  1. @Test
  2. public void test4combPipelineTrans() {
  3. jedis = new Jedis("localhost");
  4. long start = System.currentTimeMillis();
  5. Pipeline pipeline = jedis.pipelined();
  6. pipeline.multi();
  7. for (int i = 0; i < 100000; i++) {
  8. pipeline.set("" + i, "" + i);
  9. }
  10. pipeline.exec();
  11. List<Object> results = pipeline.syncAndReturnAll();
  12. long end = System.currentTimeMillis();
  13. System.out.println("Pipelined transaction: " + ((end - start)/1000.0) + " seconds");
  14. jedis.disconnect();
  15. }

但是经测试(见本文后续部分),发现其效率和单独使用事务差不多,甚至还略微差点。

5、分布式直连同步调用

  1. @Test
  2. public void test5shardNormal() {
  3. List<JedisShardInfo> shards = Arrays.asList(
  4. new JedisShardInfo("localhost",6379),
  5. new JedisShardInfo("localhost",6380));
  6.  
  7. ShardedJedis sharding = new ShardedJedis(shards);
  8.  
  9. long start = System.currentTimeMillis();
  10. for (int i = 0; i < 100000; i++) {
  11. String result = sharding.set("sn" + i, "n" + i);
  12. }
  13. long end = System.currentTimeMillis();
  14. System.out.println("Simple@Sharing SET: " + ((end - start)/1000.0) + " seconds");
  15.  
  16. sharding.disconnect();
  17. }

这个是分布式直接连接,并且是同步调用,每步执行都返回执行结果。类似地,还有异步管道调用

6、分布式直连异步调用

  1. @Test
  2. public void test6shardpipelined() {
  3. List<JedisShardInfo> shards = Arrays.asList(
  4. new JedisShardInfo("localhost",6379),
  5. new JedisShardInfo("localhost",6380));
  6.  
  7. ShardedJedis sharding = new ShardedJedis(shards);
  8.  
  9. ShardedJedisPipeline pipeline = sharding.pipelined();
  10. long start = System.currentTimeMillis();
  11. for (int i = 0; i < 100000; i++) {
  12. pipeline.set("sp" + i, "p" + i);
  13. }
  14. List<Object> results = pipeline.syncAndReturnAll();
  15. long end = System.currentTimeMillis();
  16. System.out.println("Pipelined@Sharing SET: " + ((end - start)/1000.0) + " seconds");
  17.  
  18. sharding.disconnect();
  19. }

7、分布式连接池同步调用

如果,你的分布式调用代码是运行在线程中,那么上面两个直连调用方式就不合适了,因为直连方式是非线程安全的,这个时候,你就必须选择连接池调用

  1. @Test
  2. public void test7shardSimplePool() {
  3. List<JedisShardInfo> shards = Arrays.asList(
  4. new JedisShardInfo("localhost",6379),
  5. new JedisShardInfo("localhost",6380));
  6.  
  7. ShardedJedisPool pool = new ShardedJedisPool(new JedisPoolConfig(), shards);
  8.  
  9. ShardedJedis one = pool.getResource();
  10.  
  11. long start = System.currentTimeMillis();
  12. for (int i = 0; i < 100000; i++) {
  13. String result = one.set("spn" + i, "n" + i);
  14. }
  15. long end = System.currentTimeMillis();
  16. pool.returnResource(one);
  17. System.out.println("Simple@Pool SET: " + ((end - start)/1000.0) + " seconds");
  18.  
  19. pool.destroy();
  20. }

上面是同步方式,当然还有异步方式

8、分布式连接池异步调用

  1. @Test
  2. public void test8shardPipelinedPool() {
  3. List<JedisShardInfo> shards = Arrays.asList(
  4. new JedisShardInfo("localhost",6379),
  5. new JedisShardInfo("localhost",6380));
  6.  
  7. ShardedJedisPool pool = new ShardedJedisPool(new JedisPoolConfig(), shards);
  8.  
  9. ShardedJedis one = pool.getResource();
  10.  
  11. ShardedJedisPipeline pipeline = one.pipelined();
  12.  
  13. long start = System.currentTimeMillis();
  14. for (int i = 0; i < 100000; i++) {
  15. pipeline.set("sppn" + i, "n" + i);
  16. }
  17. List<Object> results = pipeline.syncAndReturnAll();
  18. long end = System.currentTimeMillis();
  19. pool.returnResource(one);
  20. System.out.println("Pipelined@Pool SET: " + ((end - start)/1000.0) + " seconds");
  21. pool.destroy();
  22. }

需要注意的地方:

1)事务和管道都是异步模式。在事务和管道中不能同步查询结果。比如下面两个调用,都是不允许的,如下:

  1. Transaction tx = jedis.multi();
  2. for (int i = 0; i < 100000; i++) {
  3. tx.set("t" + i, "t" + i);
  4. }
  5. System.out.println(tx.get("t1000").get()); //不允许
  6. List<Object> results = tx.exec();


  7. Pipeline pipeline = jedis.pipelined();
  8. long start = System.currentTimeMillis();
  9. for (int i = 0; i < 100000; i++) {
  10. pipeline.set("p" + i, "p" + i);
  11. }
  12. System.out.println(pipeline.get("p1000").get()); //不允许
  13. List<Object> results = pipeline.syncAndReturnAll();

2)事务和管道都是异步的,个人感觉,在管道中再进行事务调用,没有必要,不如直接进行事务模式。

3)分布式中,连接池的性能比直连的性能略好(见后续测试部分)。

4)分布式调用中不支持事务,因为事务是在服务器端实现,而在分布式中,每批次的调用对象都可能访问不同的机器,所以,没法进行事务。

转载自:http://www.open-open.com/lib/view/open1410485827242.html

9、redis之事务2-Jedis的八种调用方式(事务、管道、分布式)介绍的更多相关文章

  1. 【转载】Redis的Java客户端Jedis的八种调用方式(事务、管道、分布式…)介绍

    转载地址:http://blog.csdn.net/truong/article/details/46711045 关键字:Redis的Java客户端Jedis的八种调用方式(事务.管道.分布式…)介 ...

  2. Redis的Java客户端Jedis的八种调用方式(事务、管道、分布式)介绍

    jedis是一个著名的key-value存储系统,而作为其官方推荐的java版客户端jedis也非常强大和稳定,支持事务.管道及有jedis自身实现的分布式. 在这里对jedis关于事务.管道和分布式 ...

  3. Jedis的八种调用方式(功能:事务,管道)

        1. packagecom.irwin.redis;     2.       3. importjava.util.Arrays;     4. importjava.util.List; ...

  4. Java客户端Jedis的八种调用方式

      redis是一个著名的key-value存储系统,而作为其官方推荐的java版客户端jedis也非常强大和稳定,支持事务.管道及有jedis自身实现的分布式. 在这里对jedis关于事务.管道和分 ...

  5. Spring事务Transaction配置的五种注入方式详解

    Spring事务Transaction配置的五种注入方式详解 前段时间对Spring的事务配置做了比较深入的研究,在此之间对Spring的事务配置虽说也配置过,但是一直没有一个清楚的认识.通过这次的学 ...

  6. Spring事务之详解--三种实现方式

    实现购买股票案例: 一.引入JAR文件: 二.开始搭建分层架构---创建账户(Account)和股票(Stock)实体类 Account: /* * 账户 */ public class Accoun ...

  7. Selenium Webdriver元素定位的八种常用方式

    楼主原创,欢迎学习和交流,码字不容易,转载请注明出处,谢谢. 在使用selenium webdriver进行元素定位时,通常使用findElement或findElements方法结合By类返回的元素 ...

  8. Selenium Webdriver元素定位的八种常用方式(转载)

    转载自 https://www.cnblogs.com/qingchunjun/p/4208159.html 在使用selenium webdriver进行元素定位时,通常使用findElement或 ...

  9. 爬虫-【selenium—Webdriver元素定位的八种常用方式

    在使用selenium webdriver进行元素定位时,通常使用findElement或findElements方法结合By类返回的元素句柄来定位元素.其中By类的常用定位方式共八种,现分别介绍如下 ...

随机推荐

  1. Orchard模块开发全接触2:新建 ProductPart

    一:创建 Part 1:项目引用 Orchard.Framework: 2:创建 Models 文件夹: 3:在 Models 文件夹下创建类 ProductPartRecord,如下: public ...

  2. 在LaTeX中使用颜色 Using colours in LaTeX

    Using colours in LaTeX There are several elements in LATEX whose colour can be changed to improve th ...

  3. 图像质量评估(IQA)

    图像质量评估函数的分类曾是一个比较有争议的话题,在2l世纪以前曾经有过 比较多的讨论.但是随着研究的深入和技术的广泛应用,研究人员对于图像质量 评估函数的分类有了统一的认识,即从实际应用中参考信息供给 ...

  4. 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 ...

  5. ECC校验

    ECC的全称是 Error Checking and Correction or Error correction Coding,是一种用于差错检测和修正的算法.NAND闪存在生产和使用中都会产生坏块 ...

  6. golang常用模块介绍

    golang模块 一.命令行库Cobra Cobra提供简单的接口来创建强大的现代化CLI接口,比如git与go工具.Cobra同时也是一个程序, 用于创建CLI程序 https://www.jian ...

  7. jQuery中attr()和prop()的区别,修改checked属性 jquery attr('checked' 不起作用 attr('checked' 不对

    在做复选框全选按钮的时候,出现了一个问题,使用语句$.attr('checked',true),将复选框的属性改为被选中,在chrome浏览器中第一次点击有效后面就不行了,IE8倒是没有问题. 百度了 ...

  8. 【BZOJ】【3339】Rmq Problem

    离线+线段树 Orz Hzwer,引用题解: 这一题在线似乎比较麻烦 至于离线.. 首先按照左端点将询问排序 然后一般可以这样考虑 首先如何得到1-i的sg值呢 这个可以一开始扫一遍完成 接着考虑l- ...

  9. 生成学习算法(Generative Learning algorithms)

    一.引言 前面我们谈论到的算法都是在给定\(x\)的情况下直接对\(p(y|x;\theta)\)进行建模.例如,逻辑回归利用\(h_\theta(x)=g(\theta^T x)\)对\(p(y|x ...

  10. 7.5 zookeeper客户端curator的基本使用 + zkui

    使用zookeeper原生API实现一些复杂的东西比较麻烦.所以,出现了两款比较好的开源客户端,对zookeeper的原生API进行了包装:zkClient和curator.后者是Netflix出版的 ...