这里写目录标题
- 1.IO(阻塞IO)和NIO(非阻塞IO)的概念
- 2.redis的线程模型
- 3.redis的通信协议
1.IO(阻塞IO)和NIO(非阻塞IO)的概念
1.jdk1.4引入了NIO,但也有很多人在用阻塞IO,这两者有什么区别呢?假设线程1现在要从服务端读取1个10个字节的数据,但是只读到了5个字节,没有读完,那么线程1就会进入到阻塞状态。NIO就是线程1即使只读到了5个字节也会直接将这读到的5个字节返回,不会阻塞,等剩下5个字节加载进来后,再去读取。
2.redis的线程模型
1.redis虽然是基于单线程的,但是其效率高,是为什么呢?
2.原因:
1)redis中所有数据都是基于内存的,所有的计算都是内存级别的计算,所以很快
2)redis在处理并发的客户端连接时,使用的是NIO(非阻塞IO),那么它是如何通过非阻塞的方式来不断地读取数据的呢,其实这里要提一个多路复用这个概念,本质是一个时间轮询的API,过一段时间就不断地来读取数据
3)redis会给每一个客户端指令通过队列来进行排队处理
4)rredis做出响应时,也会有一个响应队列
5)redis是单线程的,所以一些时间复杂度高的指令,比如keys,可能会出现卡顿现象
3.redis的通信协议
1.redis使用的通信协议为文本协议,这个文本协议叫Redis Serialization Protocol 简称RESP。Redis协议将传输的结构分为五种最小的单元,单元结束时,加上\r\n
2.单行字符串以 + 开始,例如+hellojava\r\n
3.多行字符串以$开头,后面加上字符串的长度,例如$4java\r\n\
4.整数值以:开头,例如:25
5.数组以*开头,后面加上数组的长度
- 执行redis中的set和get命令
1)先在redis的配置文件注释掉需要密码验证这一项,还有关闭保护模式
2)定义socket连接redis,并且提供set和get方法
package com.yl;import java.io.IOException;
import java.io.StringReader;
import java.net.Socket;public class RedisProtocolClient {private Socket socket;public RedisProtocolClient () {try {socket = new Socket("192.168.244.129",6379);} catch (IOException e) {e.printStackTrace();}}//定义redis的set命令,传一个数组到redis[set,key,value]public String set(String key,String value) throws IOException{StringBuffer sb = new StringBuffer();sb.append("*3").append("\r\n").append("$").append("set".length()).append("\r\n").append("set").append("\r\n").append("$").append(key.getBytes().length).append("\r\n").append(key).append("\r\n").append("$").append(value.getBytes().length).append("\r\n").append(value).append("\r\n");socket.getOutputStream().write(sb.toString().getBytes());byte[] buff = new byte[1024];socket.getInputStream().read(buff);return new String(buff);}//定义get方法.[get,key]public String get(String key) throws IOException{StringBuffer sb = new StringBuffer();sb.append("*2").append("\r\n").append("$").append("get".length()).append("\r\n").append("get").append("\r\n").append("$").append(key.getBytes().length).append("\r\n").append(key).append("\r\n");socket.getOutputStream().write(sb.toString().getBytes());byte[] buff = new byte[1024];socket.getInputStream().read(buff);return new String(buff);}
}
3)测试
package com.yl;import java.io.IOException;public class RedisProtocolTest {public static void main(String[] args) throws IOException {String s1 = new RedisProtocolClient().set("k1","hellojava");System.out.println(s1);String s2 = new RedisProtocolClient().get("k1");System.out.println(s2);}
}
4)结果