Redis管道简单使用及与普通方式效率对比
2017-07-26 17:31
411 查看
请求/响应协议和RTT
Redis是一种基于客户端-服务端模型以及请求/响应协议的TCP服务。这意味着通常情况下一个请求会遵循以下步骤:
客户端向服务端发送一个查询请求,并监听Socket返回,通常是以阻塞模式,等待服务端响应。
服务端处理命令,并将结果返回给客户端。
因此,例如下面是4个命令序列执行情况:
Client: INCR X Server: 1 Client: INCR X Server: 2 Client: INCR X Server: 3 Client: INCR X Server: 41
2
3
4
5
6
7
8
1
2
3
4
5
6
7
8
[/code]
客户端和服务器通过网络进行连接。这个连接可以很快(loopback接口)或很慢(建立了一个多次跳转的网络连接)。无论网络延如何延时,数据包总是能从客户端到达服务器,并从服务器返回数据回复客户端。
这个时间被称之为 RTT (Round Trip Time - 往返时间). 当客户端需要在一个批处理中执行多次请求时很容易看到这是如何影响性能的(例如添加许多元素到同一个list,或者用很多Keys填充数据库)。例如,如果RTT时间是250毫秒(在一个很慢的连接下),即使服务器每秒能处理100k的请求数,我们每秒最多也只能处理4个请求。
如果采用loopback接口,RTT就短得多,但它任然是一笔很多的开销在一次批量写入操作中。幸运的是有一种方法可以改善这种情况。
Redis 管道(Pipelining)
一次请求/响应服务器能实现处理新的请求即使旧的请求还未被响应。这样就可以将多个命令发送到服务器,而不用等待回复,最后在一个步骤中读取该答复。这就是管道(pipelining),是一种几十年来广泛使用的技术。例如许多POP3协议已经实现支持这个功能,大大加快了从服务器下载新邮件的过程。
redis很早就支持管道(pipelining)技术,因此无论你运行的是什么版本,你都可以使用管道(pipelining)操作Redis。下面是一个使用的例子:
$ (printf "PING\r\nPING\r\nPING\r\n"; sleep 1) | nc localhost 6379 +PONG +PONG +PONG1
2
3
4
1
2
3
4
[/code]
这一次我们没有为每个命令都花费了RTT开销,而是只用了一个命令的开销时间。
非常明确的,用管道顺序操作的第一个例子如下:
Client: INCR X Client: INCR X Client: INCR X Client: INCR X Server: 1 Server: 2 Server: 3 Server: 41
2
3
4
5
6
7
8
1
2
3
4
5
6
7
8
[/code]
重要说明: 使用管道发送命令是,服务器将被迫回复一个队列答复,占用很多内存。所以,如果你需要发送大量的命令,最好是把他们按照合理数量分批次的处理,例如10K的命令,读回复,然后再发送另一个10k的命令,等等。这样速度几乎是相同的,但是在回复这10k命令队列需要非常大量的内存用来组织返回数据内容。
测试
使用jedis测试普通方式和管道方式的效率差别:public class RedisPipel { public static void main(String[] args) { common(); pipeline(); } public static void pipeline() { Jedis jedis = new Jedis("ip地址",6379); jedis.auth("123456"); Pipeline pipeline = jedis.pipelined(); long start = System.currentTimeMillis(); for(int i=0; i<1000; i++){ pipeline.set("p" + i, "value" + i); } List<Object> results = pipeline.syncAndReturnAll(); long end = System.currentTimeMillis(); System.out.println("Pipelined cost time : " +((end-start)/1000.0) + " seconds"); jedis.disconnect(); } public static void common(){ Jedis jedis = new Jedis("ip地址",6379); jedis.auth("123456"); long start = System.currentTimeMillis(); for(int i=0; i<1000; i++){ jedis.set("p" + i, "value" + i); } long end = System.currentTimeMillis(); System.out.println("common cost time : " +((end-start)/1000.0) + " seconds"); jedis.disconnect(); } }1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
[/code]
在局域网情况下测试,开启了管道操作后,往返时延已经被改善得相当低,在1000次连续操作的情况下,管道的效率已经是非管道的244倍,如果操作次数的进一步增加,效率会持续增加。
common cost time : 33.017 seconds Pipelined cost time : 0.135 seconds1
2
1
2
[/code]
管道(Pipelining) VS 脚本(Scripting)
大量 pipeline 应用场景可通过 Redis 脚本(Redis 版本 >= 2.6)得到更高效的处理,后者在服务器端执行大量工作。脚本的一大优势是可通过最小的延迟读写数据,让读、计算、写等操作变得非常快(pipeline 在这种情况下不能使用,因为客户端在写命令前需要读命令返回的结果)。应用程序有时可能在 pipeline 中发送 EVAL 或 EVALSHA 命令。Redis 通过 SCRIPT LOAD 命令(保证 EVALSHA 成功被调用)明确支持这种情况。
相关文章推荐
- Redis管道简单使用及与普通方式效率对比
- Redis客户端连接方式Hiredis简单封装使用,连接池、屏蔽连接细节
- Redis中Value使用hash类型的效率是普通String的两倍 - Gabriel Zhang
- Redis中Value使用hash类型的效率是普通String的两倍
- redis简单使用及用JSON字符串的方式解决对象存储问题
- Redis和Memcached在功能、性能以及内存使用效率上的对比
- PHP静态延迟绑定和普通静态效率简单对比
- 普通物理表、临时表和TABLE(函数)的执行效率对比
- 生成大量随机字符串不同实现方式的效率对比
- 生成大量随机字符串不同实现方式的效率对比
- 生成大量随机字符串不同实现方式的效率对比
- 使用hibernate前后的效率对比
- 使用命名管道的OVERLAPPED方式实现非阻塞模式编程
- 通过编码方式使用性能计数器来进行性能计数的一个简单例子 - ZT
- [ASP.NET2.0]Repeater C# 分页用法的效率及简单优化(不使用存储过程)
- 小技巧--用简单的脚本和快捷方式提升工作效率
- Redis 集群搭建和简单使用教程
- Spring集成JSF的最简单方式是使用DelegatingVariableResolver类
- 普通ajax与使用微软ajax1.0控件对比
- 一个简单的系统配置文件使用DTD和XSD两种方式描述