jedis实现分布式锁

通过此篇文章可以了解Redis的底层通信,Redis的协议,以及自己手写与服务器通信.

在分布式锁的实现上, 基于Redis的实现是其中一种.

而具体的实现依赖包又有两个

<dependency>
    <groupId>redis.clients</groupId>
    <artifactId>jedis</artifactId>
    <version>3.0.1</version>
</dependency>

<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.10.1</version>
</dependency>

本篇文章我们就讲解第一种Jedis.

基于Jedis又有两种实现

// 单机
Jedis(String host, int port, int connectionTimeout, int soTimeout)

// 集群
JedisCluster(Set<HostAndPort> jedisClusterNode, int timeout, int maxAttempts, GenericObjectPoolConfig poolConfig)

不管是单机还是集群,它们的底层和服务器之间的通信都是基于java.net.Socket

比如说我们通过Jedis(String host, int port, int connectionTimeout, int soTimeout)构造了一个客户端,连接单机的服务器.

public Jedis(final String host, final int port, final int connectionTimeout, final int soTimeout) {
    // 调用父类BinaryJedis
    super(host, port, connectionTimeout, soTimeout);
}
public BinaryJedis(final String host, final int port, final int connectionTimeout,
      final int soTimeout) {
    // 创建一个Client,它并没有连接服务器,只是先保存了host,port. 在Client类内部有个Socket属性.
    client = new Client(host, port);
    client.setConnectionTimeout(connectionTimeout);
    client.setSoTimeout(soTimeout);
}

调用首次调用setnx向服务器发送命令时,会连接服务器

public Long setnx(final String key, final String value) {
    checkIsInMultiOrPipeline();
    // 调用上面构造好的client的setnx方法
    client.setnx(key, value);
    return client.getIntegerReply();
  }

最后会跟进到如下代码

protected Connection sendCommand(final Command cmd, final byte[]... args) {
    try {
        // 连接服务器
        connect();
        // 发送指令
        Protocol.sendCommand(outputStream, cmd, args);
        pipelinedCommands++;
        return this;
    } catch (JedisConnectionException ex) {
        // ...
    }
}

跟进到connect()方法

public void connect() {
    // 判断socket是否已连接,如果没有连接服务器则连接服务器
    if (!isConnected()) {
        try {
            socket = new Socket();
            socket.setReuseAddress(true);
            socket.setKeepAlive(true);
            socket.setTcpNoDelay(true);
            socket.setSoLinger(true, 0);
            socket.connect(new InetSocketAddress(host, port), connectionTimeout);
            socket.setSoTimeout(soTimeout);
            // 输出流
            outputStream = new RedisOutputStream(socket.getOutputStream());
            // 输入流
            inputStream = new RedisInputStream(socket.getInputStream());
        } catch (IOException ex) {
            broken = true;
            throw new JedisConnectionException(ex);
        }
    }
}

这就是最熟悉的通过Socket和服务器进行通信,还有输入输出流.

连接好服务器之后,接下来就是发送命令给服务器了.

跟进到发送命令的代码

private static void sendCommand(final RedisOutputStream os, final byte[] command,
      final byte[]... args) {
    // 通过输出流向服务器发送数据
    try {
        os.write(ASTERISK_BYTE);
        os.writeIntCrLf(args.length + 1);
        os.write(DOLLAR_BYTE);
        os.writeIntCrLf(command.length);
        os.write(command);
        os.writeCrLf();

        for (final byte[] arg : args) {
            os.write(DOLLAR_BYTE);
            os.writeIntCrLf(arg.length);
            os.write(arg);
            os.writeCrLf();
        }
    } catch (IOException e) {
        throw new JedisConnectionException(e);
    }
}

通过输出流,将命令发送给服务器.

下面我们将这个方法里面的常量替换一下,把一些不必要的代码删除,再看下这个方法

// #1
os.write('*');
os.write(args.length + 1);
os.write("\r\n");

// #2
os.write('$');
os.write(command.length);
os.write("\r\n");
os.write(command);
os.write("\r\n");

// #3
for (final byte[] arg : args) {
    os.write('$');
    os.write(arg.length);
    os.write("\r\n");
    os.write(arg);
    os.write("\r\n");
}

在说这段代码之前,我们要说下Redis的协议.

互联网的通信是基于协议的,我们熟悉的TCP/IP协议,Dubbo通信的dubbo协议,Zookeeper的zookeeper协议,RocketMQ通信的自身应用层协议.没有协议,那么客户端和服务器就不能通信,彼此'听不懂'对方在说什么.

那么Redis客户端和服务器之间要想彼此知道对方说的什么,那么它们之间也有通过协议通信,这就是Redis协议.

具体协议如下

*<number of arguments> CR LF
$<number of bytes of argument 1> CR LF
<argument data> CR LF
...
$<number of bytes of argument N> CR LF
<argument data> CR LF

比如我们要向服务器发送SET mykey myvalue这个命令,那么转换成协议之后,具体的内容如下

*3
$3
SET
$5
mykey
$7
myvalue

并不是我不把它们写成一行,而是在它们彼此之间有'\r\n',也就是回车换行. 我们简单介绍下它

*3

*这个符号是固定的,那么后面这个3是什么意思呢,3表示后面有3个内容(或者说命令由3部分组成),细心的读者也看到了,整个命令中有3个符号,每个符号代表一个内容.

$3
SET

$符号是固定的,后面的3表示后面有3个字符,因为SET就是3个字符组成的

$5
mykey

$符号是固定的,后面的5表示后面有5个字符,因为mykey就是5个字符组成的

$7
myvalue

$符号是固定的,后面的7表示后面有7个字符,因为myvalue就是7个字符组成的

协议的编解码也是一个话题. Redis的协议是基于长度的,通过长度就可以准确的知道,命令的开始在哪里,结束又在哪里. 基于长度的协议有很多,比如Dubbo或者RocketMQ的协议,它们将数据发送给对方之后,对方就是通过基于长度的解码器,将数据解码出来.

相信读者朋友应该明白了Redis的协议.那么我们只要通过Socket的输出流将这些协议内容发送给服务器就可以了,服务器基于协议,就能'读懂'我们发送给它的命令是什么了.

我们再回到上面的那段代码.

// #1
os.write('*');
os.write(args.length + 1);
os.write("\r\n");

// #2
os.write('$');
os.write(command.length);
os.write("\r\n");
os.write(command);
os.write("\r\n");

// #3
for (final byte[] arg : args) {
    os.write('$');
    os.write(arg.length);
    os.write("\r\n");
    os.write(arg);
    os.write("\r\n");
}

相信这个时候,你再来看这段代码应该就明白了. 就是平铺直叙的将协议'翻译'成代码.

所以说,当我们需要和服务器通信的时候,也未必是必须依赖Redis的依赖包,我们完成可以自己通过Socket与服务器直接通信. 比如下面这段简单的代码,就可以直接和Redis通信了,当然它很简单.这里只是提供给你一个思路.

import java.io.IOException;
import java.net.Socket;

public static void connectRedis(String key, String value) throws IOException {

    Socket client = new Socket(host, port);

    // 执行 set key value命令
    StringBuilder command = new StringBuilder();

    String number = "*3" + CRLF;
    command.append(number);

    String cmd = "$3" + CRLF + "SET" + CRLF;
    command.append(cmd);

    cmd = "$" + key.getBytes().length + CRLF + key + CRLF;
    command.append(cmd);

    cmd = "$" + value.getBytes().length + CRLF + value + CRLF;
    command.append(cmd);

    // 向服务器发送命令
    client.getOutputStream().write(command.toString().getBytes());

    // 接收服务器响应
    byte[] response = new byte[1024];
    client.getInputStream().read(response);
    System.out.println(new String(response, 0, response.length));

}

在开篇我们也讲到,Redis分布式锁的实现有jedis和redisson两种. jedis的底层通信是直接基于Socket, 而redisson的底层与服务器通信是基于Netty.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351