热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

phpredis如何使用pipeline,redis中pipeline详解

一、pipeline出现的背景:redis执行一条命令有四个过程:发送命令、命令排队、命令执行、返回结果;这个过程称为Roundtript

655184f0cf06e0fe53db341a0a647370.png

一、pipeline出现的背景:

redis执行一条命令有四个过程:发送命令、命令排队、命令执行、返回结果;

这个过程称为Round trip time(简称RTT, 往返时间),mget mset有效节约了RTT,但大部分命令(如hgetall,并没有mhgetall)不支持批量操作,需要消耗N次RTT ,这个时候需要pipeline来解决这个问题。

二、pepeline的性能

1、未使用pipeline执行N条命令

cbf295f0170108cd2143b0b2aa5f3540.png

2、使用了pipeline执行N条命令

67755fca3cf68a543963d4687e7ef641.png

3、两者性能对比

59d1a81895b1c95409be6bcf75c130f6.png

小结:这是一组统计数据出来的数据,使用Pipeline执行速度比逐条执行要快,特别是客户端与服务端的网络延迟越大,性能体能越明显。

下面贴出测试代码分析两者的性能差异:@Test

public void pipeCompare() {

Jedis redis = new Jedis("192.168.1.111", 6379);

redis.auth("12345678");//授权密码 对应redis.conf的requirepass密码

Map data = new HashMap();

redis.select(8);//使用第8个库

redis.flushDB();//清空第8个库所有数据

// hmset

long start = System.currentTimeMillis();

// 直接hmset

for (int i = 0; i <10000; i++) {

data.clear(); //清空map

data.put("k_" + i, "v_" + i);

redis.hmset("key_" + i, data); //循环执行10000条数据插入redis

}

long end = System.currentTimeMillis();

System.out.println(" 共插入:[" + redis.dbSize() + "]条 .. ");

System.out.println("1,未使用PIPE批量设值耗时" + (end - start) / 1000 + "秒..");

redis.select(8);

redis.flushDB();

// 使用pipeline hmset

Pipeline pipe = redis.pipelined();

start = System.currentTimeMillis();

//

for (int i = 0; i <10000; i++) {

data.clear();

data.put("k_" + i, "v_" + i);

pipe.hmset("key_" + i, data); //将值封装到PIPE对象,此时并未执行,还停留在客户端

}

pipe.sync(); //将封装后的PIPE一次性发给redis

end = System.currentTimeMillis();

System.out.println(" PIPE共插入:[" + redis.dbSize() + "]条 .. ");

System.out.println("2,使用PIPE批量设值耗时" + (end - start) / 1000 + "秒 ..");

//--------------------------------------------------------------------------------------------------

// hmget

Set keys = redis.keys("key_*"); //将上面设值所有结果键查询出来

// 直接使用Jedis hgetall

start = System.currentTimeMillis();

Map> result = new HashMap>();

for (String key : keys) {

//此处keys根据以上的设值结果,共有10000个,循环10000次

result.put(key, redis.hgetAll(key)); //使用redis对象根据键值去取值,将结果放入result对象

}

end = System.currentTimeMillis();

System.out.println(" 共取值:[" + redis.dbSize() + "]条 .. ");

System.out.println("3,未使用PIPE批量取值耗时 " + (end - start) / 1000 + "秒 ..");

// 使用pipeline hgetall

result.clear();

start = System.currentTimeMillis();

for (String key : keys) {

pipe.hgetAll(key); //使用PIPE封装需要取值的key,此时还停留在客户端,并未真正执行查询请求

}

pipe.sync(); //提交到redis进行查询

end = System.currentTimeMillis();

System.out.println(" PIPE共取值:[" + redis.dbSize() + "]条 .. ");

System.out.println("4,使用PIPE批量取值耗时" + (end - start) / 1000 + "秒 ..");

redis.disconnect();

}

ace4be85067824aaa38e83e371761462.png

三、原生批命令(mset, mget)与Pipeline对比

1、原生批命令是原子性,pipeline是非原子性

(原子性概念:一个事务是一个不可分割的最小工作单位,要么都成功要么都失败。原子操作是指你的一个业务逻辑必须是不可拆分的. 处理一件事情要么都成功,要么都失败,原子不可拆分)

2、原生批命令一命令多个key, 但pipeline支持多命令(存在事务),非原子性

3、原生批命令是服务端实现,而pipeline需要服务端与客户端共同完成

四、Pipeline正确使用方式

使用pipeline组装的命令个数不能太多,不然数据量过大,增加客户端的等待时间,还可能造成网络阻塞,可以将大量命令的拆分多个小的pipeline命令完成。

1、Jedis中的pipeline使用方式

大家知道redis提供了mset、mget方法,但没有提供mdel方法,如果想实现,可以借助pipeline实现。

2、Jedis中的pipeline使用步骤:获取jedis对象(一般从连接池中获取)

获取jedis对象的pipeline对象

添加指令

执行指令

测试类方法:@Test

public void testCommond() {

// 工具类初始化

JedisUtils jedis = new JedisUtils("192.168.1.111", 6379, "12345678");

for (int i = 0; i <100; i++) {

// 设值

jedis.set("n" + i, String.valueOf(i));

}

System.out.println("keys from redis return =======" + jedis.keys("*"));

}

// 使用pipeline批量删除

@Test

public void testPipelineMdel() {

// 工具类初始化

JedisUtils jedis = new JedisUtils("192.168.1.111", 6379, "12345678");

List keys = new ArrayList();

for (int i = 0; i <100; i++) {

keys.add("n" + i);

}

jedis.mdel(keys);

System.out.println("after mdel the redis return ---------" + jedis.keys("*"));

}

JedisUtils下的mdel方法:/**

* 删除多个字符串key 并释放连接

*

* @param keys*

* @return 成功返回value 失败返回null

*/

public boolean mdel(List keys) {

Jedis jedis = null;

boolean flag = false;

try {

jedis = pool.getResource();//从连接借用Jedis对象

Pipeline pipe = jedis.pipelined();//获取jedis对象的pipeline对象

for(String key:keys){

pipe.del(key); //将多个key放入pipe删除指令中

}

pipe.sync(); //执行命令,完全此时pipeline对象的远程调用

flag = true;

} catch (Exception e) {

pool.returnBrokenResource(jedis);

e.printStackTrace();

} finally {

returnResource(pool, jedis);

}

return flag;

}

使用pipeline提交所有操作并返回执行结果:@Test

public void testPipelineSyncAll() {

// 工具类初始化

Jedis jedis = new Jedis("192.168.1.111", 6379);

jedis.auth("12345678");

// 获取pipeline对象

Pipeline pipe = jedis.pipelined();

pipe.multi();

pipe.set("name", "james"); // 调值

pipe.incr("age");// 自增

pipe.get("name");

pipe.discard();

// 将不同类型的操作命令合并提交,并将操作操作以list返回

List list = pipe.syncAndReturnAll();

for (Object obj : list) {

// 将操作结果打印出来

System.out.println(obj);

}

// 断开连接,释放资源

jedis.disconnect();

}

五、redis事务

pipeline是多条命令的组合,为了保证它的原子性,redis提供了简单的事务。

1、redis的简单事务,

一组需要一起执行的命令放到multi和exec两个命令之间,其中multi代表事务开始,exec代表事务结束。

30dfd09e74d22057e49a663f86ffc54a.png

2、停止事务discard

ce325eb7b45747e008add77a20a80659.png

3、命令错误,语法不正确,导致事务不能正常结束

9a9455ea105af6b6238592c90b63169a.png

4、运行错误,语法正确,但类型错误,事务可以正常结束

d7619d2641a299b38f7cf2ea81e2bc30.png

5、watch命令:

使用watch后, multi失效,事务失效

58d6b7bacaf1359094d8359cee5d9fe9.png

WATCH的机制是:在事务EXEC命令执行时,Redis会检查被WATCH的key,只有被WATCH的key从WATCH起始时至今没有发生过变更,EXEC才会被执行。如果WATCH的key在WATCH命令到EXEC命令之间发生过变化,则EXEC命令会返回失败。

更多redis知识请关注redis入门教程栏目。



推荐阅读
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 模板引擎StringTemplate的使用方法和特点
    本文介绍了模板引擎StringTemplate的使用方法和特点,包括强制Model和View的分离、Lazy-Evaluation、Recursive enable等。同时,还介绍了StringTemplate语法中的属性和普通字符的使用方法,并提供了向模板填充属性的示例代码。 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • 本文介绍了Oracle数据库中tnsnames.ora文件的作用和配置方法。tnsnames.ora文件在数据库启动过程中会被读取,用于解析LOCAL_LISTENER,并且与侦听无关。文章还提供了配置LOCAL_LISTENER和1522端口的示例,并展示了listener.ora文件的内容。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
  • Java程序设计第4周学习总结及注释应用的开发笔记
    本文由编程笔记#小编为大家整理,主要介绍了201521123087《Java程序设计》第4周学习总结相关的知识,包括注释的应用和使用类的注释与方法的注释进行注释的方法,并在Eclipse中查看。摘要内容大约为150字,提供了一定的参考价值。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
author-avatar
dgh
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有